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进入 21 世纪 , 生活 和 工作 的 快 节奏 令 我 们 目不暇接 , 各 种 各 样 的 信息 充斥 着 我 们 的 视野 、 
撞击 着 我 们 的 思维 。 追 忆 过 去 ，Windows 操作 系统 的 诞生 成 就 了 微软 的 霸主 地 位 ， 也 造就 了 
PC 时 代 的 繁荣 。 而 以 Android 和 iPhone 手机 为 代表 的 智能 移动 设备 的 广泛 应 用 则 标志 着 移动 
互联 网 时 代 〈3G 时 代 ) 已 经 来 临 ， 谁 将 会 成 为 这 些 移动 设备 上 的 主 罕 ? 目前 看 来 ， 最 有 可 能 
的 就 是 Android PC 时 代 的 Windows! 


U 360000000 


随 着 3G 时 代 的 到 来 ， 无 线 带 宽 越 来 越 高 ， 使 得 更 多 精彩 的 应 用 程序 可 以 在 手机 上 运行 ， 
如 视频 通话 、 视 频 点 播 、 移 动 互 联网 冲浪 、 在 线 看 书 / 听 歌 及 内 容 分 享 等 。 为 了 承载 这 些 数据 
,手机 功能 越 来 越 智能 , 越 来 越 开 放 。 这 就 需要 有 一 个 好 的 开发 平台 来 支持 , 而 由 Google 
司 发 起 的 OHA 联盟 走 在 了 业界 的 前 列 ， 于 2007 年 11 月 推出 了 开放 的 Android 平台 ,任何 
司 及 个 人 都 可 以 免费 获取 到 源 代码 及 开发 包 。 由 于 Android 的 开放 性 和 优异 的 性 能 ， 得 到 
了 业界 广泛 的 支持 ， 其 中 包括 各 大 手机 厂商 和 著名 的 移动 运营 商 。 自 2008 年 9 月 第 一 款 基 了 
Android 平台 的 手机 G1 发 布 之 后 ,三星 、 摩 托 罗 拉 、 索 爱 、LG、 华 为 等 公司 都 推出 了 Android 
平台 手机 ， 中 国 移动 也 联合 各 手机 广 商 共同 推出 基于 Android 平台 的 OPhone。 
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从 技术 角度 而 言 ，Android 与 让 hone 相似 ， 采 用 WebKit 浏览 器 引擎 ， 具备 触 摸 屏 、 高 级 
图 形 显示 和 上 网 功能 , 用 户 能 够 在 手机 上 查收 电子 邮件 、 搜 索 网 址 和 观看 视频 节目 等 Android 
手机 比 Phone 等 其 他 手机 更 强调 搜索 功能 ， 界面 更 强大 ,可 以 说 是 一 种 融入 了 全 部 Web 应 用 
的 平台 。 随 着 版 本 的 更 新 ， 从 最 初 的 触 屏 到 现在 的 多 点 触摸 ， 从 普通 的 联系 人 到 现在 的 数据 
同步 ， 从 简单 的 谷歌 地 图 到 现在 的 导航 系统 ， 从 基本 的 网 页 浏览 到 现在 的 HIML 5， 这 都 说 
明 Android 已 经 逐渐 稳定 ， 而 且 功能 越 来 越 强 大 。 此 外 ，Android 平台 不 仅 文 持 Java、C、C++ 
等 主流 的 编程 语言 ， 还 支持 Ruby、Python 等 脚本 语言 。Google 甚至 专 为 Android 的 应 用 开发 
推出 了 Simple 语言 ， 这 使 得 Android 有 着 非常 广泛 的 开发 群体 。 
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本 书 循序 渐进 地 讲解 了 Android 技术 的 基本 知识 , 并 通过 实例 的 方式 介绍 了 Android 在 各 
个 领域 的 具体 应 用 。 本 书 内 容 新 颖 、 知 识 全 面 、 讲 解 详细 。 全 书 分 为 10 章 ， 其 中 1、2 章 是 
基础 知识 ， 讲 解 搭 建 开发 环境 的 过 程 和 基础 知识 ， 第 3 章 详 细 讲 解 了 Android 在 人 机 交互 界 
面 领域 典型 实例 的 设计 过 程 ， 第 4 章 讲解 了 Android 各 个 组 件 的 使 用 过 程 ; 第 5 章 讲解 了 
Android 在 交互 式 应 用 领域 的 实现 过 程 ; 第 6 章 讲 解 了 Android 在 手机 自动 服务 领域 中 的 实现 
过 程 ， 第 7 章 讲解 了 Android 在 娱乐 多 媒体 领域 中 的 具体 实现 过 程 ， 第 8 章 讲解 了 Android 
在 互联 网 领域 各 个 范例 的 实现 过 程 ， 第 9 章 讲 解 了 Android 在 官方 服务 绑 定 领 域 各 个 范例 的 


Ce 


依从 避 


人 


实现 过 程 ， 第 10 章 是 典型 手机 游戏 开发 应 用 。 书 中 每 个 范例 先 提出 制作 思路 及 包含 知识 点 ， 


在 实例 最 后 补充 总 结 知识 点 ， 让 读者 举一反三 。 
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本 书 内 容 相当 丰富 ， 实 例 覆 盖 面 广 ， 满 足 Android 技术 人 员 成 长 通路 上 的 方方面面 。 作 


以 完善 自己 的 知识 和 技能 结构 。 在 内 容 的 编写 上 ， 本 书 具 有 以 下 特色 。 
(1) 结构 合理 


者 的 目标 是 通过 一 本 图 书 ， 提 供 多 本 图 书 的 价值 ， 读 者 可 以 根据 自己 的 需要 有 选择 地 阅读 ， 


从 用 户 的 实际 需要 出 发 ， 科 学 安排 知识 结构 ， 内 容 由 浅 入 深 ， 人 叙述 清楚 ， 并 附 有 相应 的 


总 结 和 练习 ， 具 有 很 强 的 实用 性 ， 反 映 了 当前 Android 技术 的 发 展 和 应 用 水 平 。 同 时 全 书 精 


心得 选 的 最 具 代 表 性 、 读 者 最 关心 的 知识 点 ， 几 乎 包括 Android 技术 的 各 个 方面 。 


(2) 易学 易 懂 


成 体系 ， 使 读者 既 可 以 按照 本 书 编排 的 章节 顺序 进行 学 习 ， 也 可 以 根据 
节 有 针对 性 地 学 习 。 
(3) 实用 性 强 


本 书 彻底 气 弃 枯燥 的 理论 和 简单 的 操作 ， 注 重 实 用 性 和 可 操作 性 ， 讲 解 了 各 个 实例 的 具 


本 书 条 理 清晰 、 语 言 简 洁 ， 可 帮助 读者 快速 掌握 每 个 知识 点 ， 每 个 部 分 既 相互 连贯 又 自 


自己 的 需求 对 某 一 章 


体 实 现 过 程 ， 使 用 户 掌握 相关 操作 技能 的 同时 ， 还 能 学 习 到 相应 的 理论 知识 。 


(4) 实例 全 面 


书 中 的 开发 实例 都 很 典型 并 具有 创意 ， 涵 盖 了 Android 所 涉及 的 所 有 和 领域， 每 个 实例 都 


~ 


体现 了 移动 互联 网 应 用 所 需 的 创新 精神 及 良好 的 用 户 体 验 理念 
思考 和 学 习 。 


这 个 设计 思路 很 值得 大 家 去 
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0 10 00AndroduUUuo 


Android 的 发 


二 


为 了 使 读者 能 够 更 好 地 学 习 本 书 ， 在 本 节 的 内 容 


展 历 


= 和 = 月 
月 叶 ， 


程 和 让 读者 了 解 Android 的 发 展 之 路 。 


1.1.1 智能 手机 的 定义 
所 谓 智 


扩充 ， 并 可 以 通过 移动 通信 网 络 来 实现 无 线 网 络 接 入 的 这 样 


能 手机 〈SmartPhone)， 是 指 “ 像 个 人 电脑 一 相 
户 自行 安 疲 软件、 游戏 等 第 三 方 服务 商 提 供 的 程序 ， 通 过 | 


智能 手机 的 基本 知识 ， 为 读者 学 习 本 章 后 面 的 内 容 打 好 基础 。 


FE， 具 有 独立 的 操作 系统 ， 可 以 由 用 


PP， 将 首先 讲解 和 Android 关系 密切 的 


一 类 手机 的 总 


称 ”。 简单 


能 手机 就 是 


部 像 电 脑 一 样 可 以 通过 下 载 和 安装 软件 来 拓展 其 功能 的 手机 。 


此 类 程序 来 不 断 对 手机 的 功能 进行 
地 说 ， 知 


智能 手机 可 以 是 传统 的 手机 增加 智能 功能 , 例如 Symbian 操作 系统 的 S60 系列 、Windows 


Mobile 操作 系统 的 Windows Mobile Smartphone 系列 ; 也 可 以 是 传统 PDA 加 上 手机 通信 功能 ， 
例如 Windows Mobile 操作 系统 的 Windows Mobile Pocket PCPhone 系列 、Palm 操作 系统 的 
Treo 系列 ; 也 可 以 是 其 他 独立 类 型 ， 例 如 Symbian 操作 系统 的 S80、UIQ， 以 及 一 些 Linux 


操作 系统 的 智能 手机 。 然 而 ， 就 新 近 的 发 展 来 看 ， 这 些 智 


能 手机 的 类 型 有 相 融 


合 的 趋势 。 


“智能 手机 (SmartPhone )” 这 个 说 法 主要 是 针对 “功能 手机 (Featurephone )” 而 来 的 ， 
本 身 并 不 意味 着 这 个 手机 有 多 “智能 (Smart)” 从 另 一 个 角度 来 讲 ， 所 谓 的 “智能 手机 


(SmartPhone )” 就 是 一 
( Featurephone ) ”是 不 能 随 
(Featurephone )” 具 备 了 安装 Java 应 | 


台 可 以 随 


尽 女 


总 E 
意 安装 


装 


不 


和 弛 载 应 用 软件 的 手机 就 像 电 脑 那 样 )。“ 功 


能 手机 


1 印 载 软件 的 ，Java 的 出 现 使 后 来 的 “功能 手机 


程序 的 功能 ， 但 是 Java 程序 的 操作 友好 性 、 


以 及 对 系统 资源 的 操作 都 比 “ 智 能 手机 〈SmartPhone)” 差 很 多 。 


1.1.2 


智能 手机 的 特点 


三 | 
口 
口 


口 


Ley 


浏览 


口 


4 备 


< 人 


智能 手机 的 主要 特点 如 下 。 
备 普通 手机 的 全 部 功能 ， 能 够 进行 正常 的 通话 、 短 信 等 应 用 
备 无 线 接 入 互联 网 的 能 力 , 即 需要 支持 GSM 网 络 下 的 GPRS 
CDMA 1X 或 者 3G 网 络 。 
L 备 PDA 的 功能 ， 如 PIM 个 人 信息 管理 )、 日 程 记 事 、 任 务 安排 、 多 媒体 应 
具有 天 


F 放 性 的 操作 系统 , 在 这 个 操作 系统 平台 上 ， 


或 者 CDMA 网 


可 以 安 半 更 多 的 应 月 


运行 效率 


络 下 的 


用 以 及 


一 Android 开发 完全 实战 宝典 
从 而 使 智能 手机 的 功能 可 以 得 到 扩充 。 


1.1.3 “主流 智能 手机 系统 


当今 世界 中 比较 著名 的 手机 操作 系统 有 如 下 几 种 。 
1. Symbian: Symbian OS( 中 文 译音 “ 塞 班 系 统 ”) 
Symbian 是 由 诺基亚 、 索 尼 爱 立信 、 靡 托 罗 拉 、 西 门 子 等 


1L 家 大 型 移动 通讯 设备 商 


同 


出 资 组 建 的 一 个 合资 公司 ， 专 门 研发 手机 操作 系统 ， 现 已 被 诺基亚 全 资 收 购 。Symbian 很 


是 Windows 和 Linux 的 结合 体 ， 有 着 良好 的 界面 ， 采 用 内 核 与 


而 设计 。 
2. Windows Phone 


像 


界面 分 离 技 术 ， 对 硬件 的 要 求 
比较 低 ， 支 持 CHt+、VB 和 J2ME， 兼 容 性 较 差 。 目 前 根据 人 机 界面 的 不 同 ，Symbian 体系 的 
用 户 界面 (User Interface，UI) 平台 分 为 Series 60、Series 80、Series 90、UIQ 等 。Series 60 
主要 是 为 数字 键盘 手机 而 设计 ，Series 80 是 为 完整 键盘 而 设计 ，Series 90 则 是 为 触 控 笔 方式 


Windows Phone 是 微软 发 布 的 一 款 手机 操作 系统 , 它 将 微软 旗下 的 Xbox Live 游戏 .Zune 


音乐 与 独特 的 视频 体验 整合 至 手机 中 。2010 年 10 月 11 日 晚上 9 点 30 分， 微软 公司 正式 发 


布 了 智能 手机 操作 系统 Windows Phone。 2011 年 2 月 , 诺基亚 与 微软 达成 全 球 战略 同盟 并 深 


度 合作 共同 研发 。2012 年 3 月 21 日 ，Windows Phone 7.5 登 问 


二 中国。6 月 21 日 ， 微 软 正式 


发 布 最 新 手机 操作 系统 Windows Phone 8，Windows Phone 8 将 采用 和 Windows 8 相同 的 


内 核 。 
Windows Phone 具有 桌面 定制 、 图 标 拖 搜 、 滑 动 控 制 等 一 


系列 前 卫 的 操作 体验 。 其 主 


上 FE 屏 


幕 通过 提供 类 似 仪表 盘 的 体验 来 显示 新 的 电子 邮件 、 短 信 、 未 接 来 ! 


对 重要 信息 保持 时 刻 更 新 。 它 还 包括 一 个 增强 的 触摸 屏 界面 ， 


3. Linux 


昌 、 日历 约会 等 ， 让 人 们 


更 方便 手指 操作 ;，Windows 
Phone， 力 图 打破 人 们 与 信息 和 应 用 之 间 的 隔 疼 ， 提 供给 人 们 最 优秀 的 端 到 端 体验 。 


Linux 是 源 于 PC 的 移动 操作 系统 ， 具 有 上 面 2 个 操作 系统 无 法 比拟 的 优势 : 其 一 ，Linux 


具有 开放 的 源 代码 ， 能 够 大 大 降低 开发 成 本 ;其 二 ，Linux 既 满 
有 针对 性 地 开发 自己 的 Linux 手机 操作 系统 的 要 求 ， 又 吸引 ] 


足 了 手机 制造 商 根据 实际 情况 


众多 软件 开发 商 对 内 容 应 月 


件 的 开发 ， 丰 富 了 第 三 方 应 用 。 然而 Linux 操作 系统 有 其 先天 


的 不 足 : 入 门 难度 大 、 熟 悉 其 
开发 环境 的 工程 师 少 、 集 成 开发 环境 较 差 ; 由 于 微软 PC 操作 系统 源 代 码 的 不 公开 , 基 了 


日 软 


Linux 


的 产品 与 PC 的 连接 性 较 差 ; 尽管 目前 从 事 Linux 操作 系统 开发 的 公司 数量 较 多 , 但 真正 具有 


很 强 开发 实力 的 公司 却 很 少 ， 而 且 这 些 公司 之 间 是 处 于 相互 独立 的 开发 状态 ， 很 难 实现 更 大 


的 技术 突破 。 最 初 摩托 罗拉 非常 推 案 Linux 平台 ， 然 而 在 和 诺基亚 的 较量 中 不 断 失 败 ， 现 在 
也 不 再 那么 热心 Linux 了 ， 转 而 投向 基于 Linux 的 Android 平台 ， 其 


关注 。 
4. BlackBerry 


出 的 Android 手机 很 受 


“黑莓 ”(BlackBerry) 是 加 拿 大 RIM 公司 推出 的 一 种 移动 电子 邮件 系统 终端 ， 其 特色 是 
支持 推送 式 电子 邮件 、 手 提 电 话 、 文 字 短 信 、 互 联网 传真 、 网 页 浏览 及 其 他 无 线 资讯 服务 。 
黑 稚 最 强大 也 是 最 有 优势 的 方面 在 于 收发 邮件 ,然而 在 中 国 ， 用 手机 收发 邮件 还 不 是 很 流 


所 以 黑 芍 在 中 国 几 乎 没有 多 大 市 场 。 
2 国 目 


行 ， 


5. iOS 


iOS 是 苹果 公司 (Apple. Inc.) 公司 手机 产品 iPhone 专 ) 


公司 首席 执行 官 史 蒂 夫 。 乔 布 斯 在 2007 年 
29 日 在 美国 上 市 。iPhone 将 创 间 
页 浏览 、 搜 索 和 地 图 功能 的 因特网 ; 


1 月 9 
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的 智能 手机 系统 。iPhone 由 苹果 


举行 的 Macworld 宣布 推出 ，2007 年 6 月 
FE、 可 触摸 宽屏 iPod 以 及 具有 桌面 级 电子 邮件 、 网 


这 三 利 


iPhone“iPhone 是 一 款 苹 命 履 


大 型 多 触 点 显示 屏 和 领先 性 新 软件 的 全 
还 开创 了 移动 设备 软件 尖端 功能 的 
的 、 


了 五 年 。” 苹 果 公司 首 订 执 行 官 史 蒂 夫 。 乔 布 


新 说 : “手指 是 


产品 完美 地 融 为 一 体 。iPhone 引入 了 基 了 
] 户 界面 ， 让 用 户 用 手指 即 可 控制 iPhone。iPhone 
重新 定义 了 移动 电话 的 功能 。 有 人 这 样 评价 
不 可 思议 的 产品 ， 比 市 场 上 的 其 他 任何 移动 电话 整整 领先 
们 与 生 俱 来 的 终极 定点 设备 ， 


而 iPhone 利用 它们 创造 了 自 鼠 标 以 来 最 具 创 新 意义 的 


6. Android 


Android 一 词 的 本 义 指 “机 器 人 ” 同时 也 是 Google 公司 于 2007 年 11 月 5 日 宣布 的 基 ] 
Linux 平台 的 开源 手机 操作 系统 的 名 称 ， 该 
成 ， 号 称 是 首 个 为 移动 终端 打造 的 真 了 


区 会 


] 户 界 


看 ?9 
日 。 


已 经 成 为 市 场 占 有 率 最 高 的 智能 手机 操作 系统 。 


12 [0 0 Android 


操作 系统 、 
E 开 放 和 完整 的 移动 软件 。 


P 间 件 、 用 户 界 面 和 应 用 软件 组 


研 止 本 书 成 形 之 时 ，Android 


Android 平台 采用 了 WebKit 浏览 器 引擎 ， 具 备 触摸 屏 、 高 级 图 形 显示 和 上 网 功能 ， 用 户 


能 够 在 手机 上 查看 电子 邮件 、 搜 索 网 址 


并 观看 视频 节目 ， 


同时 Android 还 具有 比 iPhone 等 其 


他 手机 更 强 的 搜索 功能 ， 可 以 说 是 一 种 融入 全 部 Web 应 用 的 平台 。 


1.2.1 产生 背景 


Android 是 Google 公司 开发 的 基于 Linux 平台 的 开源 手机 操作 系统 。Google 与 开放 手机 


联盟 合作 开发 了 Android， 这 个 联盟 计 
T-Mobile 在 内 的 30 多 家 技术 和 无 线 应 | 
商 、 开 发 商 和 其 他 有 关 各 方 结 成 深层 次 的 合作 伙 作 
动 电话 软件 平台 ， 在 移动 产业 内 形成 一 个 玫 
开放 手机 联盟 的 成 立 和 Android 的 


包括 中 国 


移动 、 摩 托 罗拉 、 高 通 、 宏 达 电 (HTC) 和 
的 领军 企业 组 成 。Google 通过 与 运营 商 、 设 备 制造 
EF 关系 ， 希 望 借助 建立 标准 化 、 开 放 式 的 移 


F 放 式 的 生态 系统 。 


E 出 是 对 现状 的 


E 大 改变 ， 在 和 带 来 初步 效益 之 前 ， 还 


需要 不 小 的 耐心 和 高 昂 的 投入 。 但 是 ， 笔 者 认为 如 果 意 识 到 全 球 移动 用 户 从 中 能 获得 的 潜在 


利益 ， 是 值得 付出 这 些 努 力 的 。 


1.2.2 Android 手 机 介绍 


2008 年 9 月 22 日 ， 美 国运 营 商 工 Mobile USA 在 纽约 正式 发 布 第 一 款 Google 手机 一 一 


工 Mobile G1。 该 款 手 机 为 中 国 
的 手机 ， 支 持 WCDMA/HSPA 


摩托 罗拉 的 首 款 Android 手机 CLIQ， 如 


台湾 宏达电 代 工 制造 ， 是 世界 上 第 一 部 使 用 Android 操作 系统 
里 论 下 载 速率 7.2Mbit/s， 并 支持 Wi-Fi。 
图 1-1 所 示 。 


搭载 Google Android 2.0 的 摩托 罗拉 Moto Droid 如 图 1-2 所 示 。 


画 国 3 


Android 开发 完全 实战 宝典 
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We Rr” Ps 
We Ee 
| 


Android 的 主要 特性 如 下 。 


口 


应 用 程序 框架 ， 


支持 组 建 的 3 


1-1 摩托 罗拉 的 首 款 Android 手机 


芭 用 与 蔡 换 。 


Dalvik 虚拟 机 ， 专 门 为 移动 设备 做 了 优化 。 


内 部 集成 浏览 


[a ， 该 浏览 旷 


和 基于 


图 1-2 搭载 Google Android 2.0 的 
摩托 罗拉 Moto Droid 


开源 的 WebKit 引擎 。 


优化 的 图 形 库 , 包括 2D 和 3D 图 形 库 , 3D 图 形 库 基于 OpenGL ES 1.0 (硬件 加 速 可 选 )。 
#SQLite， 用 做 结构 化 的 数据 存储 。 


多 媒体 文 持 ， 包 括 常 见 的 音频 、 视 频 和 静 ; 


AAC、 AMR、JPG、PNG、GIF)。 
GSM 电话 (依赖 于 人 硬件 )。 


蓝牙 Bluetooth、EDGE、3G 和 WiFi (依赖 于 硬件 )。 
GPS、 指 南 针 和 加 速度 计 〈 依 赖 于 硬件 )。 

丰富 的 开发 环境 ， 包 扣 
集成 开 


照相 机 、 


发 环境 插件 。 


5 设备 模拟 器 、 
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和 主流 的 开发 工具 一 


样 , Android 也 有 自 


组 件 结构 应 用 程序 框架 的 基本 知识 
Android 组 件 结构 


1.4.1 


Android 采用 了 软件 


基础 ， 


4 罩 画 


太 影 


UN 


己 的 组 伯 


性 能 


F, 在 本 节 的 内 容 中 , 将 简 


只 提供 基本 功能 ， 其 他 的 应 用 软件 则 由 各 公司 自行 开发 ， 


一 部 分 。 


分 析 图 表 ， 


像 文件 格式 (如 MPEG4、H.264、MP3、 


以 及 Eclipse 


要 介绍 Android 


堆 层 (Software Stack， 又 名 软件 琶 层 ) 的 架构 ， 低 层 以 Linux 内 核 为 


以 Java 作为 编写 程序 的 
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1.4.2 ”Android 应 用 程序 框架 


Android 会 同一 个 核心 应 用 程序 包 一 起 发 布 ， 该 应 月 


消息 程序 、 日 历 、 地 I、 


软件 的 重用 ; 从 


用 户 替换 。 


以 下 所 有 的 应 | 


口 一 个 可 扩 


口 内 容 管理 


上 程序 包 包括 


浏览 器 及 联系 人 管理 等 程序 。 所 有 的 应 
开发 者 也 完全 可 以 访问 核心 应 用 程序 所 使 用 的 API 框架 。 
人 应 用 程序 都 可 以 发 布 它 的 功能 块 ， 并 
用 其 所 发 布 的 功能 块 〈 不 过 得 遵循 框架 的 安全 性 限制 )。 该 应 有 


程序 都 由 一 系列 的 服务 和 系统 组 成 。 
(Views): 可 以 用 来 建立 应 ) 


Email 客户 端 、 


该 应 用 程序 架构 


程序 都 是 月 


日 Java 编写 的 。 


任何 其 
程序 重 


文本 框 (Text Boxes)、 按 钮 (Buttons)， 甚 至 包括 
器 〈Content Providers ): 使 得 应 用 程序 可 以 访问 男 一 个 应 | 


口 一 个 资源 管理 


壹 息 


百 , 蔬 \。 


Android 系统 提供 给 应 


口 一 个 通知 管理 


口 一 个 活动 类 管理 
航 回 退 功能 。 


一 个 Android 程序 编译 运行 后 的 效果 如 图 1-3 所 示 。 


联系 人 数据 库 )， 或 者 共享 它们 自己 的 数据 。 
器 〈Resource Manager): 提供 非 代码 资源 的 访问 ， 如 本 地 字符 上 
和 分 层 文 件 (Layout Files)。 


器 《Notification Manager): 使 得 应 用 程序 可 以 在 状态 术 


程序 都 可 
组 伯 


也 的 应 
了 机 


] 来 简化 组 件 
以 使 
F 可 以 被 


程序 , 包括 列表 (Lists)、 网 格 (Grids)、 
个 可 秽 入 的 浏览 器 。 
程序 的 数据 (如 


器 〈Activity Manager): 用 来 管理 应 用 
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图 1-3 ”Android 程序 运行 效果 


在 开发 应 用 时 就 是 在 这 个 框架 上 进行 扩展 ， 下 面 来 看 Android 框架 都 有 些 什么 功能 可 供 使 用 。 
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发 者 一 个 框架 ， 所 有 的 应 月 


显示 客户 通知 


SMS 短 


、 图 形 


从 癌 用 上 


男 回 应 


于 发 都 必须 遵守 这 个 框架 的 原则 。 


”Android 开发 完全 实战 宝典 

android.app: 提供 高 层 的 程序 模型 和 基本 的 运行 环境 。 
android.content: 包含 对 各 种 设备 上 的 数据 进行 访问 和 发 布 。 
android.database: 通过 内 容 提供 者 浏览 和 操作 数据 库 。 
android.graphics: 底层 的 图 形 库 ， 包 含 画 布 、 颜 色 过 滤 、 点 及 算 形 等 ， 可 以 将 它们 直 
接 绘制 到 屏幕 上 。 
android.location: 定位 和 相关 服务 的 类 。 
android.media: 提供 一 些 类 管理 多 种 音频 、 视 频 的 媒体 接口 。 

android.net: 提供 帮助 网 络 访问 的 类 ， 超 过 通常 的 java.net.* 接口 。 
android.os: 提供 了 系统 服务 、 消 息 传 输 和 IPC 机制 。 

android.opengl: 提供 OpenGL 的 工具 。 

android.provider: 提供 访问 Android 内 容 提 供 者 的 类 。 

android.telephony: 提供 与 拨打 电话 相关 的 API 交互 。 

android.view: 提供 基础 的 用 户 界面 接口 框架 。 

android.util: 涉及 工具 性 的 方法 ， 例 如 时 间 日 期 的 操作 。 

android.webkit: 默认 浏览 器 操作 接口 。 

android.widget: 包含 各 种 UI (界面 元 素 〉 元素 在 应 用 程序 的 布局 中 使 用 。 


DODDO 


DOOOOODOOODDOC 
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Android 中 提供 了 一 个 模拟 器 来 模拟 ARM 移动 设备 。 
Android 的 模拟 器 是 基于 QEMU 开发 的 ，QEMU 是 一 个 有 名 的 i 
开源 虚拟 机 项 目 〈 详 见 http://bellard.org/qemu/)， 它 可 以 提供 一 — 
虚拟 的 ARM 移动 设备 。 开 发 人 员 不 需要 一 个 真实 的 手机 ， 只 
需 通 过 电脑 即 可 模拟 运行 手机 操作 系统 , 即 可 开发 出 应 用 在 手机 
面 程序 。 模 拟 器 在 电脑 上 模拟 运行 的 效果 如 图 1-4 所 示 。 
在 本 节 的 内 容 中 ， 将 简要 介绍 Android 模拟 器 的 基本 知识 。 
1.5.1 Android 模 拟 器 介绍 

对 于 Android 程序 的 开发 者 来 说 , 模拟 器 的 推出 给 开发 者 在 
开发 和 测试 上 带 来 了 很 大 的 便利 。 无 论 在 Windows 下 还 是 Linux 
下 ，Android 模拟 器 都 可 以 顺利 运行 ， 而 且 官 方 提供 了 Eclipse 
插件 ， 可 将 模拟 器 集成 到 Eclipse 的 IDE 环境 。 当 然 ， 也 可 以 从 
命令 行 启动 Android 模拟 器 。 

获取 模拟 器 的 方法 非常 简单 ， 既 可 以 从 官方 站 点 
(http://developer.Android.com/) 免费 下 载 单 独 的 模拟 器 ， 也 可 以 在 下 载 Android SDK 后 ， 解 压 后 
在 其 SDK 的 根 目录 下 找到 一 个 名 为 “tools” 文 件 夹 ， 此 文件 夹 下 包含 了 完整 的 模拟 器 和 一 些 
非常 有 用 的 工具 。 
Android SDK 中 包含 的 模拟 器 的 功能 非常 齐全 ， 电 话 本 、 通 话 等 功能 都 可 正常 使 用 ， 甚 
至 其 内 置 的 浏览 器 和 Maps 都 可 以 联网 。 用 户 可 以 使 用 键盘 输入 ， 鼠 标 单 击 模拟 器 按键 输入 ， 
6 国 面 


> 


1 


图 1-4 模拟 器 模拟 手机 
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也 可 以 使 用 鼠标 单 击 、 拖 动 屏幕 进行 操纵 。 


1.5.2 ”模拟 器 和 真 机 的 区 别 
Android 模拟 器 和 真 机 的 不 同 之 处 如 下 。 
口 不 文 持 呼 叫 和 接听 实际 来 电 ， 但 可 以 通过 控制 台 模 拟 电话 呼叫 〈 呼 入 和 呼出 )。 
口 不 文 持 USB 连接 。 
口 不 文 持 相 机 /视频 捕捉 。 
口 不 文 持 音 频 输 入 《捕捉 )， 但 支持 输出 《〈 重 放 )。 
口 不 文 持 扩展 耳机 。 
口 不 能 确定 连接 状态 。 
口 不 能 确定 电池 电量 水 平和 交流 充电 状态 。 
口 不 能 确定 SD 卡 的 插入 /弹出 。 
口 不 文 持 蓝牙 。 
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老牌 智能 手机 软件 平台 制造 商 Symbian 发 言 人 表示 ，Google 的 Android 只 不 过 是 男 一 个 
Linux，Symbian 对 其 他 软件 与 其 形成 的 竞争 并 不 感到 担心 。 除 了 北美 之 外 ，Symbian 在 其 它 
地 区 智能 手机 市 场 都 占有 大 部 分 市 场 份额 。 但 是 事实 是 从 2011 年 第 一 季度 开始 到 现在 ， 
Android 已 经 成 为 市 场 占 有 率 最 高 的 智能 手机 操作 系统 。 


1.6.1 Android 的 未 来 发 展 

与 让 hone 相似 ，Android 采用 WebKit 浏览 器 引擎 ， 具备 触摸 屏 、 高 级 图 形 显示 和 上 网 功 
能 ， 用 户 能 够 在 手机 上 查看 电子 邮件 、 搜 索 网 址 和 观看 视频 节目 等 ， 比 iPhone 等 其 他 手机 更 
强调 搜索 功能 ， 界 面 更 强大 ， 可 以 说 是 一 种 融入 全 部 Web 应 用 的 单一 平台 。 

但 其 最 震撼 人 心 之 处 在 于 Android 手机 系统 的 开放 性 和 免费 服务 。Android 是 一 个 对 第 三 
方 软件 完全 开放 的 平台 ， 开 发 者 在 为 其 开发 程序 时 拥有 更 大 的 自由 度 ， 突 破 了 iPhone 等 只 能 
添加 固定 软件 的 柳 锁 ; 同时 与 Windows Mobile、Symbian 等 厂商 不 同 ，Android 操作 系统 免费 
向 开发 人 员 提 供 ， 这 样 可 节省 近 三 成 成 本 。 

Android 项 目 己 经 从 手机 运营 商 、 手 机 三 商 、 开 发 者 和 消费 者 那里 获得 大 力 文 持 。Google 
移动 平台 主管 安 迪 。 篆 宾 (Andy Rubin) 表示 ， 与 软件 开发 合作 伙伴 的 密切 接触 正在 进行 中 。 
在 过 去 的 四 年 间 , 市 场 上 产品 种 类 最 多 的 智能 手机 产品 便 是 Android。 几乎 涵盖 了 所 有 的 品牌 ， 
各 种 不 同 的 配置 。 
1.6.2 Android 的 市 场 前 景 

2011 年 初 ， 仅 正式 推出 三 年 的 Android 已 超越 称霸 十 年 的 Symbian 系统 ， 采 用 Android 
系统 的 重要 厂商 包括 美国 Motorola、 韩 国 Samsung、 英 国 Sony Ericsson; 男 外 中 国 厂商 如 : 
HTC、 联 想 、 华 为 、 中 兴 等 ， 使 之 跃 居 全 球 最 受 欢 迎 的 智能 手机 平台 ，Android 系统 不 但 利用 
于 智能 手机 ， 也 在 平板 电脑 市 场 急速 扩大 。 
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“ 工 欲 善 其 事 ， 必 有 


SF 
要 


个 合适 


的 工具 


Android 开发 环境 的 基本 知识 。 


E 利 其 器 ”这 名 名言 出 
。 对 于 编程 人 员 来 说 ， 开 发 了 
效 的 开发 工具 后 ， 搭 建 一 个 开发 平台 则 是 首要 的 人 


1 
导 心 契 


月 《论语 》， 
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作为 一 门 


新 兴 技 术 ， 在 进行 3 


境 前 ， 需 要 了 


建 Android 开发 环境 的 准备 工作 。 


说 要 想 高 效 地 完成 一 件 事 ， 


[ 具 至 关 重 要 。 选 择 了 Android 这 样 一 个 高 
F 务 。 在 本 章 的 内 容 中 ， 将 讨 


F 细 介绍 搭建 


TF 发 前 ， 首 先 要 搭建 一 个 对 应 的 
音 安 装 开发 工具 所 需要 的 便 件 和 软件 配置 条 件 。 在 本 节 内 容 中 ， 将 首先 讲解 拱 


发 环境 。 而 在 搭建 开 发 环 


2.1.1 基本 系统 要 求 
开发 基于 Android 的 应 用 软件 所 需要 的 开发 环境 如 表 2-1 所 示 。 
表 2-1 开发 系统 所 需 参数 
项 版 本 要 求 说 明 备 注 
是 作 去 和 Windows XP /Vista/7 Mac OS X 1 、 4 一、 OE 总 采用 坊 作 受 妨 
操作 系统 10.4.8+Linux Ubuntu Drapper 根据 自己 的 电脑 自行 选择 选择 自己 最 熟悉 的 操作 系统 
软件 开发 包 Android SDK 选择 最 新 版 本 的 SDK 
Eclipse3.3 (Europa), 3.4 
IDE Eclipse IDE+ADT (Ganymede) 和 ADT(Android 选择 “for Java Developer” 
Development Tools) 开 发 插件 
Java SE Development Kit5 或 6 (单独 的 JRE 不 可 以 的 ， 必 须要 
其 他 JDK Apache Ant Linux 和 Mac 上 使 用 Apache Ant | 有 JDK) ， 不 兼容 Gnu Java 编 
1.6.5+, Windows 上 使 用 1.7+ 版 本 译 器 (gcj) 
2.1.2 Android 软 件 开发 包 
Android 的 软件 开发 包 主 要 包括 以 下 工具 。 


下 载 。 


口 配套 的 


安装 Android 的 Eclipse 捐 


发 插件 。 


口 JDK: 可 以 到 网 址 http:/www.oracle.com/technetwork/java/javase/downloads/index.html 


口 Eclipse (Europa): 可 以 到 网 址 http:/www.eclipse.org/downloads/ 下 载 Eclipse IDE for 
Java Developers 。 


口 Android SDK: 可 以 到 网 址 http://developer.android.com 下 载 。 


件 需 注意 ，Eclipse 可 以 到 对 应 的 网 站 下 载 安 装 ， 如 果 通 过 网 络 
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远程 安装 不 成 功 ， 可 以 下 载 到 本 地 安装 。 


2.2 Windowsl DODDODO 


六 


Android 开发 环境 的 基本 知识 。 


2.2.1 安装 JDK、Eclipse、Android SDK 


因为 当前 主流 的 操作 系统 是 Windows， 所 以 在 此 将 首先 详细 计 


} 解 在 Windows 中 


a 
I 
也 


下 面具 体 介 绍 JDK 1.5、Eclipse 3.3、ADT1.5、Android SDK 的 安装 步骤 ， 在 配套 的 视频 
中 也 有 详细 的 介绍 。 


1. 安装 JDK 


第 1 步 : 安装 Eclipse 的 开发 环境 需要 JRE 的 文 持 ， 在 Windows 上 安装 JRENJDK 非 


常 简单 ， 


首先 在 Oracle 官方 网 站 下 载 ， 网 址 为 http://www.oracle.com/technetwork/java/javase/downloads/ 
index.html， 如 图 2-1 所 示 。 


NE Downloads Top Downloads 


From enterprise software to developer tools, Sun offers a comprehensive suite of products that help to 
create solutions and increase productivity. The latest releases are available for download below. 


To see the list of all downloadable software from Sun, please visitthe Sun Download Center 


口 Database 日 Infrastructure Software 
a > = 


e Connectivity (JDBC) 


jects (JDO) 二 
nce AP| 4# 


国 = More Info 


Fish Enterprise Server # 
5 日 二 


习 Java Card 

由 Java EE gTechnologies 
习 Java ME 向 Solaris 

由 Java SE 和 由 Virtualization 

习 JavaFX 由 Web Services 


第 2 步 : 在 图 


图 2-1 Sun 官方 下 载 页 面 


2-1 中 可 以 看 到 有 很 多 版 本 ， 运 行 Eclipse 时 虽然 只 需要 JRE 就 可 以 了 , 但 


是 在 开发 Andriod 应 用 程序 的 时 候 ， 需 要 完整 的 JDK (JDK 已 经 包含 了 JRE)， 且 要 求 其 版 本 


在 1.5+ 以 上 ， 这 里 选择 Java SE QJDK) 6， 其 下 载 页 面 如 图 2-2 所 示 。 
bavaPlatform, StandardEdition | 


阅 


DK 6 Update 22 (JDK or JRE) 
his release includes performance improvements and 
Security vulnerability fixes. Learn more » 


at Java Do | Need? You must have a copy of the JRE 
Java Runtime Environment on your system to run Java 
applications and applets. To develop Java applications 
and applets, you need the JDK (Java Development Kib， 
hich includes the JRE 


Dowmload J vx| 


Dowmload 了 FE| 


了 ReleaseNotes 


HDK 6 Docs HRE 6 Docs 

了 Installation 了 Installation 
Instructions Instructions 

Fr ReadMe Fr ReadMe 


了 ReleaseNotes 


Fr Dracle License 


了 Dracle License 


了 Third Party 


Licenses 


Fr Supported System 


『 Third Party 


Licenses 


Supported System 


Configurations 


Configurations 


图 2-2 JDK 下载 页 再 
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第 3 步 : 在 图 2-2 中 选择 “JDK 6 Update 22 (JDK or JRE)”， 单 击 其 右 侧 的 “Download” 
按钮 ， 出 现 让 用 户 选 择 其 操作 系统 和 语言 的 界面 ， 在 此 首先 选择 “Windows”， 然后 单 击 


“Download” 按 钮 ， 如 图 2-3 所 示 。 


JDK 6 Update 17 

Download Java SE Development Kit 6u17 This special release provides a few key fixes. 
Platform: 
Windows 了 | 
Language: 
Muti-language eleas 

» Sun Lic 
By selecting 'Download' or 'Continue' below, you Third P 
hereby accept the terms and conditions of the Java 
SE Development Ki 6u17 License Agreement. Supported 


厂 Use Sun Download Manager (Learn More) 
Download * 
Your download will begin shortly, please 
Wait… 


Click here if your download did not start 
automatically 


图 2-3 选择 “Windows” 


第 4 步 : 经 过 上 述 操 作 后 ， 就 会 开始 下 载 安 装 文 件 “jdk-6u22-windows-i586-p.exe 
注意 : 在 此 需要 会 员 用 户 登 录 后 才能 下 载 。 


第 5 步 : 下 载 完成 后 双击 “jdk-6u22-windows-i$86-p.exe” 开 始 进行 安装 。 将 弹出 


向 导 ” 对 话 框 ， 在 此 单 击 “ 下 一 步 ”按钮 ， 如 图 2-4 所 示 。 


仿 Java (TN) SE Development Kit 6 Update 22 — i 


二 4 ORACLE 


欢迎 使 用 Javya(TM) 5E Development Kit 6 Update 22 安装 向 导 


此 向 导 将 引导 您 完成 ]ava 5E Development Kit 6 Update 22 的 安装 过 程 。 


职 消 


图 2-4 “安装 向 导 ” 对 话 框 


第 6 步 : 弹出 “安装 路 径 ” 对 话 框 ， 在 此 选择 文件 的 安装 路 径 ， 如 图 2-5 所 示 。 
第 7 步 : 单 击 “ 下 一 步 ” 按 钮 ， 开 始 进行 安装 ， 如 图 2-6 所 示 。 
第 8 步 : 完成 后 弹出 “目标 文件 夹 ” 对 话 框 ， 在 此 选择 目标 文件 夹 的 路 径 ， 如 
所 示 。 
10 国 国 
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伪 JavaCm 5E ent Kit 6 Update 22 


请 从 下 面 的 列表 中 选择 要 安装 的 可 选 功 能 。 安 装 完成 后 ， 您 可 以 使 用 "控制 面板 "中 的 ' 添 加/ 
册 除 程序 "实用 程序 来 更 改 您 选择 的 功能 


功能 说 明 


安装 到 : 
C:\Program Files\Java\jdk1.6.0_22\ 更 改 ( 有 &),,， 


< 上 一 步 (6) 取消 


[HH 

页 
tb 
ES 
江 
这 
对 
洲 


而 


2-5 “安装 路 径 ” 对 话 杠 


ava 安装 到 其 地 文件 来 


实 装 到 : 
F;\Java\jre6\ 


取消 Es 
图 2-7 “目标 文件 夹 ”对 话 # 
第 9 步 : 单 击 “下 一 步 ” 按钮 后 ， 继续 开始 安装 ， 如 图 98 所 示 。 


TH 


正在 安装 Java 
这 可 能 会 花费 几 分 钟 时 间 写 Sun 


状态 : 正在 安装 Java Runtime Environment 
下 自 因 区 间 辐 加 加 问 加 加固 则 加 因 导 加 加 轩辕 国 膨 加固 四 册 图 回 因 


| 现在 ， 您 可 以 免费 拥有 一 个 与 Microsoft Office 
| 兼容 的 功能 全 面 的 办 公 套 件 


， 一 组 功能 强大 、 集 成 在 一 起 的 字 处 理 、 电 子 表格 、 澳 示 文 稿 、 绘 图 和 数据 库 应 用 程序 | 
。 读 取 、 编 辑 和 保存 Microsoft Office 文件 

。 支持 70 多 种 语言 以 及 Solaris、Windows、Linux 和 Mac 操作 系统 

。 使 用 行业 标准 的 开放 文件 格式 {OpenDocument) 作为 默认 文件 格式 

*。 内 置 的 一 键 式 PDF 导出 


重 [| 而” 耐 [a | OpenOffioa 


mpress Base Draw Math 


图 2-8 继续 安装 
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第 10 步 : 完成 后 弹出 “完成 ”对 话 框 ， 单 击 “ 完 成 ”按钮 ， 完 成 整个 安装 过 程 ， 如 
图 2-9 所 示 。 


opment Kit 6 Update 22 - 完成 


ORACLE 


JaYa(TM) 5E Development Kit 6 Update 22 已 成 功 安装 


产品 注册 是 免费 的 ,您 将 获得 如 下 增值 服务 : 

* 获得 新 版 本 、 修 补 程序 和 更 新 的 通知 服务 

* 获 得 有 关 5un 开发 者 产品 、 服 务 和 培训 的 忧 囊 
* 获得 对 早期 版 本 和 文档 的 访问 权限 


当 您 单 击 ' 完 成 "后 将 收集 产品 与 系统 信息 ， 同 时 显示 JDK 产品 注册 表单 。 如 果 您 
不 注册 ， 则 不 保存 以 上 信息 。 


有 头 注册 所 收集 的 数据 以 及 这 些 数 据 的 管理 和 使 用 方式 的 更 多 信息 ， 请 参见 ' 产 品 
注册 信息 "页 面 。 


产品 注册 信息 (P) | 


图 2-9 完成 安装 
完成 安装 后 ， 可 以 检测 是 否 安装 成 功 ， 有 具体 的 方法 是 : 依次 单 击 “ 开 始 ” 一 “运行 ” 在 
运行 框 中 输入 “cmd” 并 按 〈Enter) 键 ， 在 打开 的 CMD 窗口 中 输入 “java -version”， 如 果 显 
示 如 图 2-10 所 示 的 提示 信息 ， 则 说 明 安 装 成 功 。 


» 


ft Windows %P [ -1.26991 
有 1985-2961 


C:\Documents and Settings dninistratorjava -version 

‘java version "1.6.8 22" 

JavaCIM> SE Runtime Environment ‘<build 1.6.9_22-bB4) 

Java HotSpot TM> Client UM Cbuila 17.1-b83, mixed mode, sharing)» 


C:\Documents and Settings fdninistrator> 


图 2-10 ”CMD 窗口 


如 果 检 测 没有 安装 成 功 ， 则 需要 将 它 的 目录 的 绝对 路 径 添 加 到 系统 的 路 径 (path〉 中 ， 
具体 方法 如 下 。 
第 1 步 : 右键 依次 单 击 “ 我 的 电脑 ”一 “属性 ”一 “高 级 ”， 单 击 下 面 的 “环境 变量 ”， 
在 下 面 的 “系统 变量 ”处 选择 新 建 在 变量 名 处 输入 “JAVA_HOME”， 变量 值 中 输入 刚才 的 目 
录 ， 例 如 “C:\Program FilesJavayjdk1.5.0_14”， 如 图 2-11 所 示 。 


I 


变 且 名 角 ): JAVA_HDNE 
变量 值 久 ): [Fr:\Tava\jdkl. 6.0_22 


ma | 


图 2-11 设置 系统 变量 
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第 2 步 : 再 新 建 一 个 变量 , 变量 名 为 classpath, 变量 值 为 “.;%JAVA_HOME%/lib/t.jar;%JAVA_ 


HOME%/lib/tools.jar”， 单 击 “ 确 定 ” 后 找到 路 径 的 变量 ， 双 击 或 单 击 编辑 ， 在 变量 值 最 前 面 
加 上 “%JAVA_HOME%/bin;”， 如 图 2-12 所 示 。 


变量 名 如) [elasspath 
变量 值 避 ) [ibyrt jar: NTAVA_ HOMES/Lib/ tools. jar 


| 


系 


= 


设 置 系统 里 


J 


图 2-12 


6 


pA 
> < 


次 依次 单 击 始 ” 运行 ” 在 运行 杠 
中 输入 “java -version”， 如 果 显 示 如 图 


输入 “cmd” 并 按 (Enter〉 键 
2-13 所 示 的 提示 信息 ， 则 说 明 


Pr 


女 


本 5.1.2688] 
3 1985-2881 Microsoft Corp- 


C:\Docunents and Settings dnministrator>java 
java version "1.6.8 22" 

JavaCIM> SE Runtime Environment build 1.6.8 22-b04> 

Java HotSpot IM> Client UM build 17?.1-hb83, mixed mode, sharing>» 


version 


C:\Documents and Settings Mdministrator> 


图 2-13 CMD 界面 


注意 : 上 述 变量 设置 中 ， 是 按照 笔者 本 人 的 安装 路 径 设 置 的 ， 笔 者 安装 JDK 的 路 
C:\Program Files\Java\jdk1.6.0_ 22. 


全 


各 


[rm 


2. 安装 Eclipse 
在 安装 好 JDK 后 ， 就 可 以 接着 安装 Eclipse 了 ， 有 具体 过 程 如 下 。 
第 1 步 : 打开 Eclipse 的 下 载 页 面 http:/www.eclipse.org/downloads/， 如 图 2-14 所 示 。 


外 


Eclipse IDE for Java EE Developers (190 MB) 
Tools for Java developers creating Java EE and VYeb applications, including a Java IDE, 
tools for Java EE, JPA, JSF, Mylyn and others. More... 

1 Downloads: 1,813,341 


Windows 32bit 

Mac Carbon 32bit 

Mac Cocoa 32bit 64bit 
Linux 32bit 64bit 


下 Enicec 


MvyECLipsE 
A 


Eclipse IDE for Java Developers (92 MB) 
The essential tools for any Java developer, including a Java IDE, a CVS client, XML Editor 
and Mylyn. More... 


Windows 32bit 
Mac Carbon 32bit 
Mac Cocoa 32bit 64bit 


Downloads: 859,165 


Eclipse IDE for C/C++ Developers (79 MB) 
An IDE for C/C++ developers with Mylyn integration. More... 
Downloads: 377,447 


ES 


第 2 步 : 在 


图 2-14 下 载 


页 再 


~ 


2-14 中 选择 “Eclipse IDE for Java Developers (92 MB)”， 在 其 


Linux 32bit 64bit 


Windows 32bit 

Mac Carbon 32bit 

Mac Cocoa 32bit 64bit 
Linux 32bit 64bit 


:下 载 的 镜像 


页 面 ， 选 择 离 用 户 最 近 的 镜像 即 可 一般 推荐 的 下 载 速度 就 不 错 )， 如 图 2-15 所 示 。 
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Home Downloads Users Members Committers Resources Projects About Us 


第 3 步 : 下 载 完 成 后 


Eclipse downloads - mirror selection 
Downloads Home ” Pp 
ps AIl downloads are provided underthe terms and conditions of the Eclipse Foundation 
Bit Torrents Software User Agreement unless otherwise specified 
好 oject 
Ey Download eclipse-java-galileo-SR1-win32 jp from: 
Bytopic 
[Chinal Amazon AWS {http} 
Ww Source code Oracle Develop 
%w More Packages @ BitTorrent ls available for this flle = ee 
-Or pick a mirror site below. 
Give Back to Come wth queetions. Leave with anewers 
本 = 和 一 一 一 


图 


2-15 选择 镜像 


,然后 找到 下 载 的 压缩 包 “eclipse-java-galileo-SR1-win32.zip”。 Eclipse 


无 须 执行 安装 程序 ， 解 压 此 所 


E 缩 文件 就 可 以 用 ， 不 过 一 定 要 先 安装 JDK。 在 此 假设 Eclipse 解 


压 后 存放 的 目录 为 “Fi\eclipse”。 
第 4 步 : 进入 解压 后 的 目录 ， 此 时 可 以 看 到 一 个 名 为 “eclipse.exe” 的 可 执行 文件 ， 双 击 
此 文件 直接 运行 ，Eclipse 能 自动 找到 用 户 先期 安装 的 JDK 路 径 ， 启 动 界面 如 图 2-16 所 示 。 


clipse contributors and others, 2000, 2009, All rights reservyed. Java and3ITJava- 
ks and logosare trademarks or registered trademarks of ,Sur Microsystems， 
., other countries, or BOtHV Eclipse is a trademark of the Eclipse Foundation, Inc. 


图 2-16 Eclipse 启动 界面 
因为 是 第 一 次 安装 、 启 动 Eclipse， 将 会 看 到 选择 工作 空间 的 提示 ， 如 图 2-17 


第 5 步 : 
所 示 。 


Select a workspace 


了 eclipse stores your projects in a folder called a workspace. 
Choose a workspace folder to use for this session. 


Workspace: 国 Neclipse\workspacel 了 ] Browse... | 


厂 Vse this as the default and do not ask again 


ca | 


图 2-17 选择 工作 空间 
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此 时 单 击 “OK” 按钮 后 ， 完 成 Eclipse 的 安装 。 系 统 进入 初始 欢迎 界面 ， 如 图 2-18 
所 示 。 


Eile Fait Noviewts Soyres Projrct Bn Bimdow Nnlp 
[ra 间 a edd 


二 Welcome to the Eclipse IDE for Java Developers 


图 2-18 初始 欢迎 界面 


3. 安装 Android SDK 

安装 JDK 和 Eclipse 完毕 后 ， 接 下 来 需要 下 载 安 装 Andriod SDK， 具 体 流程 如 下 。 

第 1 步 : 打开 Android 开发 者 社区 网 址 http://developer.android.com/， 然 后 转 到 SDK 下 载 
页 面 下 载 ， 如 图 2-19 所 示 。 


Download the Android SDK 


Welcome Developers! f you are new to the Android SDK, please read the steps below, for an overview of how to set up the SDK. 


If you're already using the Android SDK, you should update to the latest tools or platform using the Android SDK and AVD Manager. 
starter package. See Adding SDK Components. 


37448775 bytes bfbfdf8b2d0fdecc2a621544d706fa98 


installer_r18-windows exe (Recommended) 37456234 bytes 48b1fe7b431afe6b9c8a992bf75dd898 


Mac OS X (intel) android-sdk_r18-macosx.zip 33903758 bytes 8328e8a5531c9d6f6f1a0261cb97af36 
Linux (i386) android-sdk_r18-linux tgz 29731463 bytes 6cd716d0e04624b865ffed3c25b3485c 


图 2-19 SDK 下 载 页 面 


第 2 步 : 在 此 选择 用 于 Windows 平台 的 “android-sdk r18-windows.zip”， 单 击 后 弹出 下 
载 界面 ， 如 图 1-20 所 示 。 单 击 “ 保 存 文 件 ” 按 钮 ， 选 择 保存 目录 后 开始 下 载 。 
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x| 
您 已 选择 打开 


丑 android-sdk ri8-windows. zip 


为 : WinRAR ZIF 压缩 文件 (35.7 MDB) 
来 源 : http://dl. google. com 


您 想 要 Firefox 如 何 处 理 此 文件 ? 


介 打开 方式 @) [WinEAR. ZIP 性 认 】 了 | 


5 
厂 以 后 自动 采用 相同 的 动作 处 理 此 类 文件 。 忆 ) 


wm | 


图 2-20 Android SDK 下 载 页 面 


第 3 步 : 下 载 完成 后 ， 解 压 压 缩 文件 。 假 设 下 载 后 的 文件 解压 存放 在 “Fi\android\” 目 录 
下 ， 并 将 其 tools 目录 的 绝对 路 径 添 加 到 系统 的 路 径 中 ， 具 体操 作 方 法 如 下 。 

1) 用 鼠标 右键 单 击 “ 我 的 电脑 ”选择 “属性 ”一 “高 级 ” 单 击 下面 的 “环境 变量 ” 按 
钮 ， 在 下 面 的 “系统 变量 ”选项 组 处 选择 “新 建 ” 选 项 ， 在 变量 名 处 输入 “SDK_HOME?” 
变量 值 中 输入 刚才 的 目录 ， 例 如 笔者 的 就 是 “F:android-sdk-windows”， 如 图 2-21 所 示 。 


变量 名 他 ) : EDK_HonH| 
变 且 值 属 ] : [Fi vandroid-sdrwindows 


mi | 


图 2-21 设置 系统 变量 


2) 找到 PATH 的 变量 , 双击 或 单 击 编辑 , 在 变量 值 最 前 面 加 上 “%SDK_HOME%\tools;”， 
如 图 2-22 所 示 。 


变量 名 外) : [path 
变 且 值 上 : [snk HOMENNtools :WTAVA_HOMEN/bin;C:' 


| 


图 2-22 设置 系统 变量 


3) 再 次 依次 单 击 “ 开 始 ” 一 “运行 ”在 运行 框 中 输入 “cmd” 并 按 (Enter) 键 ， 在 打 
开 的 CMD 窗口 中 输入 一 个 测试 命令 ， 例 如 android-h， 如 果 显 示 如 图 2-23 所 示 的 提示 信息 ， 
则 说 明 安 装 成 功 。 
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tem32\cmd. exe 


C:\Documents and Settings\hdministrator>android -hh 


Usage: 
android [global options] action [action options] 


Global options: 

-vu -verhose Verbose mode: errors, Warnings and informational messages are pr 
inted - 

-~h ——help This help. 

S —-silent Silent mode: only errors are printed out - 


Ualid actions are Comynsnd of a verb and an optional direct object: 


一 list : Lists existing targets or virtual devices. 
list avd : Lists existing fndroid Virtual Devices. 
list target : Lists existing targets. 
create avd : Creates a new findroid Virtual Device. 
move avd : Moves or renames an findroid Virtual Device. 
delete avd : Deletes an findroid Virtual Device. 
- update avd : Updates an findroid Virtual Device to match the folders of 
a new SDK. 
create project : Creates a new findroid Project. 
— update project : Updates an findroid Project must have an findroidManifest. 
Xml> - 
- create test-phoject: Creates a new findroid Test Project. 
— update test—-project: Updates an fndroid Test Project nust have an fndroidManilb 


图 2-23 设置 系统 变量 


4. 安装 ADT 

Android 为 Eclipse 定制 了 一 个 插件 ， 即 Android Development Tools (ADT)。 这 个 插件 为 
用 户 提供 一 个 强大 的 综合 环境 用 于 开发 Android 应 用 程序 。ADT 扩展 了 Eclipse 的 功能 ， 可 以 
让 用 户 快 速 地 建立 Android 项 目 ， 创 建 应 用 程序 界面 ， 在 基于 Android 框架 API 的 基础 上 添 
加 组 件 ， 以 及 用 SDK 工具 集 调 试 应 用 程序 ， 甚 至 导出 签名 《〈 或 未 签名 ) 的 APKs 以 便 发 行 应 
用 程序 。 下 面 详细 介绍 安装 配置 ADT 的 基本 方法 。 
在 安装 ADT 插件 前 ， 需 要 首先 打开 Eclipse 集成 开发 环境 ， 接 下 来 进行 如 下 操作 。 

第 1 步 : 打开 Eclipse 后 ， 依 次 单 击 菜单 栏 的 “Help” 一 “Install New Software...”， 如 
2-24 所 示 。 

和 2 步 : 在 弹出 的 对 话 框 中 ， 单 击 “Add” 按 钮 ， 如 图 2-25 所 示 。 


Help CE 2 可 


Available Software [ 


ID 


人 Welcome Sedet site 4 snter Whe locatitn sf 4 site el 
【?】 Help Contents Troy IO 习 _ 如 | 
a we Wy rire vith Oe Mi ls SDs Hisss yr 
7) Search 

Dnanic Help Eco 

Key Assist... Ctrl+Shift+L 

Tips and Tricks... 
A Report Bug or Enhancement... i 

Cheat Sheets... 三 

厅 Show waly Un Llwit versicns of wwuilodle nmfunew 厂 京 4 itms Uat we Wrendy iastslled 

Check for Updates pe 

Install 本 了 

About Eclipse ? [mr cam 

图 2-24 添加 插件 图 2-25 单 击 “Add” 按 钮 


第 3 步 : 在 弹出 的 “Add Site” 对 话 框 中 分 别 输 入 名 字 和 地 址 ， 名 字 可 以 自己 命名 ， 例 如 
罩 国 17 
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“guan”， 但 是 “Location” 中 必须 输入 插件 的 网 络 地 址 “http://dl-ssl.google.com/Android/eclipse/”， 
单 击 “OK” 按 钮 ， 如 图 2-26 所 示 。 


二 hdd Site 


Hame: [auan Local... | 
Location: [http: /i al-ssl. google. com/ hndroid/eclipse/ Mrchive... | 


© cme | 


图 2-26 设置 地 址 


a 如 


iT 


第 4 步 : 单 击 “OK” 按钮 ， 完 成 设置 ， 此 时 在 “Install” 界 面 将 会 显示 可 用 的 插 伯 
2-27 所 示 。 


伍 Install 二 上 口 | x| 
Available Software 
Check the items that you wish to install. _ 当 
5 证 


Work with: ES - http://dl-ssl. googLe. com/Android/eclipse/ 司 上 dd | 


Find more software by working with the “Available Software Sites’ preferences. 


Hme version | 
日 明 Developer Tools 
回 堵 Mmdroid DDNS 0.9.7. v201005071157-36220 
回 只 Mndroid Development Tools 0.9.7.v201005071157-36220 


Details 
| 习 
I Show only the latest versions of available software 厂 Hide items that are already installed 
[VY Group items by category What is already installed? 


IY Contact all update sites during install to find required software 


5 or Ee ns | ee | 


图 2-27 插件 列表 


第 $ 步 : 把 “Android DDMS” 和 “Android Development Tools” 都 选中 ,然后 单 击 “Next” 
加 图 2-28 所 示 。 
6 步 : 此 时 选择 “I accept…” 选 项 ， 然 后 
所 示 。 


主意 : 上 述 步 骤 可 能 会 计算 插件 占用 资 贫 源 目 月 清 况 ， 过 程 有 点 慢 ， 读者 需要 耐心 ,小 慢 小 则 等 待 。 
完成 后 会 提示 重启 Eclipse 来 加 载 插件 ， 重 局 后 就 可 以 使 用 了 。 


单 击 “Finish” 按 钮 ， 开 始 进行 安装 ， 如 
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二 Install 


Review Licenses 


-第 2 


Licenses must be reviewed before the software can be installed. 


complete the install. 


二 


上 口 |x| 
| 
This includes licenses for software required to py S| 
Ds 


Items with licenses: 


ndroid DDMS 

最 hndroid Development Tools 
lylyn Bridge: Eclipse IDE 
lylyn Bridge: Java Development 
lylyn Bridge: Team Support 
lylyn Connector: Bugrilla 
hylyn Task List (Required) 


hylyn WikiText 


lylyn Task-Focused Interface (Rec... 


Version 


0.9.5. v200911191123-20404 


0.9.5. v200911191123-2040: 
3.3.0. v20091015-0500-e3x 
3.3.0. v20091015-0500-e3x 
3.3.0. v20091015-0500-e3x 
3.3.0. v20091015-0500-e3x 
3.3.0. v20091015-0500-e3x 
3.3.0. v20091015-0500-e3x 
1.2.0. v20091015-0500-e3x 


License text: 


Note: jcommonr1.0.12. jar is under the BSD license 国 
rather than the APL. You can find a copy of the BSD 
License at http://www. opensource. org/licenses/bsd- 
4 由 icense. php 

jfreechart-1.0.9. jar and jfreechart-1.0.9-swt. jar 
are under the LGPL rather than the AFL. You can 
find a copy of the LGPL at 
http: /www. enu. org/licenses/old-licenses/lepl- 
2.1. txt， You can get the source code for these two 
components at 
http://android. git. kernel. org/pub/ jfreechart— 
1.0.9. zip 


Apache License 
Version 2.0, January 2004 
http: /fwww. apache. org/licenses/ 


TERMS AND CONDITIONS FOR VSE, REPRODUCTION, AND 
DISTRIBUTION 本 


人 


司 CI do not accept the terms of the License agreements 


《ak | zi | aas | cc | 


项 


2-28 插 从 


安装 界面 


二 上 口 | x| 


Fetching com. android. ide. eclipse. adt 0....lipse. adt 0.9.5.v200911191123-20404. j ar 


3 Install 


加 


Fetchine com. android. ide. eclipse. a....adt_ 0.9.5.v200911191123-20404. jar 


Run in Backeround Cancel | << Details | 


开始 搭建 Android 开发 环境 


图 


另外， 不同 版 本 的 Eclipse 安装 插件 的 方法 和 步骤 是 不 同 的， 但 是 都 大 同 小 异 ， 


参照 


EENY 


上 面 的 操作 能 够 自行 解决 。 
2.2.2 设置 Android SDK Home 


安装 好 插件 后 ， 还 需要 
Android SDK 的 主 目录 。 具 体 方法 如 下 。 


所 示 。 


改 如 下 配置 才 


第 1 步 : 打开 Eclipse， 在 菜单 中 依次 单 


2-29 ”开始 安装 


相信 读者 


可 以 使 | 


] Eclipse 创建 Android 项 目 ， 需 要 设置 


击 “Windows” 一 “Preferences” 选 项 ， 如 图 2-30 
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一 


图 2-30 ”Preferences 项 


第 2 步 : 在 弹出 的 界面 左 侧 可 以 看 到 “Android” 选 项 组 ， 选 中 “Android” 选 项 后 ， 在 右 
侧 设 定 Android SDK Location 所 在 目录 ， 单 击 “OK” 按 钮 完成 安装 ， 如 图 2-31 所 示 。 


Preferences 
[sr filter text 
国 -General 


F:\android-sdk-windows 


| Target ae |Yendor |Pattorm [sr... | 


Usage Stats 
由 - 
由 - Help 
-Install/Update 
由 .Java 
由 - RuryDebug 
由 .Tasks 
由 -Team 


由 - Usage Data Collector 
Validation 
由 -XML 


图 2-31 Preferences 项 


2.23 验证 开发 环境 

经 过 本 章 前 面 知识 的 讲解 ，Android 开发 环境 搭建 完成 。 下 面 通过 新 建 一 个 项 目 来 验证 当 
前 的 环境 是 否 可 以 正常 工作 ， 具 体 流程 如 下 。 

第 1 步 : 打开 Eclipse， 在 菜单 中 依次 选择 “File” 一 “New” 一 “Project” 选 项 ， 在 弹出 
的 对 话 框 上 可 以 看 到 “Android” 选 项 ， 如 图 2-32 所 示 。 
20 国 国 加 是 


全 Wew Project 


Select a wizard 


-EE Examples 


图 2-32 新 建 项 目 


第 2 步 : 在 图 2-32 上 选择 “Android” 选 项 ， 单 击 “Next” 按 钮 , 打开 “New Android 
Project” 对 话 框 ， 在 对 应 的 文本 框 中 输入 必要 的 信息 ， 如 图 2-33 所 示 。 

第 3 步 : 这 里 不 再 具体 说 明 项 目 信息 中 各 项 的 意义 ， 后 面 章节 会 详细 介绍 。 单 击 
“Finish” 按 钮 ， 会 自动 完成 项 目的 创建 工作 ， 最 后 将 可 以 看 到 如 图 2-34 所 示 的 项 目 结构 。 


Jew Android Project 
New Android Project 


Creates a new Android Project resource. 


-ES MpiDemos 

| “名 sre 
gen [Generated Java Files] 
hndroid 2.0.1 


assets 


OC AndroidManifest. xml 
Android Open Source Project 


i 
Naroid, han Botan Prujsat 1 筷 | 
Android Open Source Project ; default. Propert1 es 
Android Dpen Source Project 

Android Dpen Source Project 

Google Inc. 

Google Inc. 

Google Ine 


Gooale Ine. ‘0. 3 gen [Generated Java Files] 
hndroid 1.1 


test123 


com. eoemobile. book. test 


GC AndroidManifest. xml 
… 有 着 default. properties 


图 2-33 “New Android Project” 对 话 框 图 2-34 项 目 结构 
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通过 上 述 操作 ， 即 可 在 Windows 平台 上 搭建 完成 开发 环境 。 


2.2.4 创建 Android AVD 虑 拟 设备 


在 Android SDK 1.5 版 以 后 的 Android 开发 中 ， 
虚拟 设备 〈Android Virtual Device)， 每 个 AVD 模拟 了 一 套 虚 拟 设 备 来 运行 Android 平台 ， 


个 平台 至 少 要 有 所 
外 观 显示 等 。 
大 


拟 和 测试 不 同 的 平台 环境 ， 创 建 AVD 的 方法 如 下 。 
1 步 : 在 CMD 下 输入 “android list targets” 


AAA 
f= 
2 


tC: WINDOW 


Nane : 


Type 
API 
Revi 
Skin 
id: 5 or 
Nane 
Type 
API 
Revi 
Skin 
id: 6 or 
Name 
Type 
API 


Revision: 


Skin 
id: ?7 or 
Nane 
Type 
API 


Revision: 


Skin 


己 的 内 核 、 系 统 图 


像 和 数据 分 区 ， 还 可 以 有 


必须 创建 至 少 一 个 AVD, AVD 即 Android 
这 
己 的 SD 卡 和 用 户 数据 以 及 


白 


为 Android SDK 1.5 以 后 文 持 多 个 平台 和 外 观 显示 ， 作 为 开发 者 创建 不 同 的 AVD 来 模 


rstem32\ cmd. exe 


Android 2.8 

: Platform 

level: 5 

sion: 1 

s: HUGA 《default>。 
"android-6" 

: findroid 2.8.1 

: Platform 

level: 6 

sion: 1 

HUGA default). 
"android-—?" 

: fndroid 2.1-updatel 
: Platform 

level: 7 

2 

HUGA 《default>。 
"android-8" 

: findroid 2.2 

: Platform 

level: 8 

‘| 

HUGA 《default>。 


QUGA . 


QUGA . 


QUGA , 


QUGA 。 


HQUGA400. 


HQUGA400. 


HQUVGA400, 


HQUGA400., 


查 


2-35 所 示 。 


看 可 用 的 平台 ， 如 图 


WQUGA432, WUGA8G0. WUGhR854 


WQUGA432, WUGAS8OG0,. WUGAS854 


WQUGA432, WUGA8G0, WUGA854 


WQUGA432, WUGA8G0, WUGA854 


C:\Documents and Settings Mdninistrator> 


如 上 列举 了 7 
第 2 步 : 创建 


格式 创建 AVD， 其 


图 2-36 所 示 。 


图 


人 


个 Android 平台 ， 


AVD。 按照 “ 


AN 


中 “your avd_name 


XP [版 本 5.1.2688] 


28061 Microsoft Corp. 


2-35 CMD 界面 


ID 分 别 是 1、2、3、4、5、6、7。 
android create avd --name <your avd name> --target <targetID>” 
”是 需要 创建 的 AVD 的 名 称 ， 在 CMD 窗口 界面 中 如 


C:\Documents and SettingsNhdministrator>android create avd 一 name mm 一 -上 target 2 
Android 1.5 is a basic hndroid platforn. 
Do vou wish to create a custom hardware profile [no] 


第 3 步 ， 这 检 


图 


2-36 CMD 界面 


就 创建 了 一 个 自 定 义 的 AVD， 然 后 ， 只 要 在 Eclipse 的 Run Configurations 


里 面 指定 一 个 虚拟 
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设 


备 ,， 即 在 Target 下 选中 


自己 定义 的 这 个 AVD, 则 sdk 1 5 _version 就 可 以 


运行 了 ， 如 图 2-37 所 示 。 
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type filter text 


Resource 
Builders 
Java Build Path 
Java Code Style 
Java Compiler 
Java Editor 
Javadoc Location 
[Project References 

RunyDebug Settings 
Task Repository 
Task Tags 
Validation 

WikiText 


Android 


Project Build Target 


口 各 troid 1.1 
口 indroid 1.5 

口 各 troid 1.6 

口 inadroid 2.0 

口 indiroid 2.0.1 

口 indroid 2.1-updatel 
口 google AFIs 

口 indroid 2.2 

Google AFIs 


Android Dpen Source Project 1 
Android Dpen Source Project 1 
Android Dpen Source Project 1 
Mndroid Dpen Source Project 2 
Android Open Source Project 2. 
Mndroid Dpen Source Froject 2. 
Google Inc. 2 
Android Dpen Source Project 2. 
Google Inc. 2 


mm 人 TAD 


-Library 
厂 Is Library 


Restore Defaults | 


Apply 


cw | 


图 2-37 选择 AVD 


第 4 步 : 选择 “sdk 1 5 _version”， 单 击 “Apply” 按 钮 后 ， 然 后 单 击 
出 “Launch” 对 话 框 ， 如 图 2-38 所 示 。 
第 5 步 : 此 时 单 击 “Launch” 按 钮 后 将 会 运行 模拟 器 ， 如 图 2-39 所 示 。 


二 Launch 0ptions 


Skin: 


Density: Medium (160) 


Sree Gl FE 


Mortor dp 
FE 


WY Wipe user data 


届 一 回 


efsnib 


图 2-38 


Cancel 


“Launch” 对 话 框 


I 


“Start” 按 钮 弹 


oOoqoO 


Ca 


hhNDROID_ 


@@O® 


hil Zl Bl il Se 
Bl WE vd 


图 2-39 模拟 运行 成 功 


关于 应 用 结构 分 析 和 讲解 ， 以 及 代码 的 调试 部 分 内 容 会 在 本 书后 面 的 内 容 中 进行 详细 介 


绍 。 至 此 ， 在 Windows 平台 上 的 开发 环境 搭建 完 


， 安 装 了 运行 环境 JDK、 玫 


[发 工具 Eclipse 


和 Android SDK， 并 安装 了 ADT， 进行 了 SDK Home 的 配置 ， 最 后 创建 了 一 个 Android 虚 


拟 设备 。 
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2 有 


除了 Windows 系统 平台 外 ，Android 开发 环境 还 可 以 在 其 他 的 操作 系统 中 搭建 。 例 如 常 
见 的 Linux 系统 和 苹果 系统 。 虽 然 系统 不 同 ， 但 是 具体 的 安装 步骤 基本 类 似 ， 为 节省 本 书 的 
篇 幅 ， 在 此 仅 做 简单 介绍 。 


2.3.1 Linux 平 台 下 的 搭建 过 程 
下 面 以 Linux Ubuntu 8.10 为 平台 ， 介 绍 搭建 Android 开发 环境 的 具体 方法 。 有 具体 流程 如 下 。 


第 1 步 : 安装 虚拟 光驱 daemon400.exe。 
第 2 步 : 在 Windows XP 下 用 虚拟 光驱 安装 Ubuntu 8.10, ISO 文件 为 ubuntu-8.10-beta-desktop- 
1386.iso。 


第 3 步 : 用 dpkg 命令 打 patch。 首 先进 入 Ubuntu 系统 ， 将 ubuntu _package 0430.tar.gz 解压 。 


tar ~zvxf ubuntu package 0430.tar.gz 


然后 打开 patch。 


sudo dpkg -i *.deb 


如 果 存在 没有 成 功 的 文件 ， 则 再 依次 执行 。 


sudo dpkg -1 filename.deb 
注意 : 可 能 有 些 需要 一 起 运行 dpkg， 具 体格 式 如 下 。 


sudo dpkg -i flenamel.deb filenamel.deb 


另外 ， 还 需要 重新 将 Java5 执行 dpkg 命令 (因为 用 Java6 会 有 问题 )。 
第 4 步 : 编译 原 码 和 Android SDK。 
编译 原 码 : 解压 原 码 到 本 地 ， 进 入 原 码 目录 ， 执 行 如 下 命令 。 


make 


编译 SDK: 在 使 用 make 命令 之 后 ， 直 接 执行 make sdk， 会 在 out/host/linux-x86/sdk 下 面 
生成 mdk 文件 及 文件 夹 ， 形 如 : android-sdk_eng.xxx_linux-x86。 

第 5 步 : 安装 Eclipse。 

只 需 直接 解压 eclipse-jee-ganymede-SR2-linux-gtk.tar.gz 即 可 。 


tar -Zzvxf eclipse-jee-ganymede-SR2-linux-gtk.tar.gz 


第 6 步 : Eclipse 下 配置 Android。 

第 7 步 : 测试 刚才 编译 好 的 SDK。 

1) 在 Eclipse 中 将 Android SDK 目录 设置 成 自己 编译 生成 的 SDK 目录 ， 例 如 
out/host/linux-x86/sdk/android-sdk eng.xxx linux-x86 。 选 择 “Window” 一 “preferences ”一 
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“Android” 选 项 中 的 SDK Location， 进 行 设 置 。 
2) 创建 AVD: 在 Eclipse 中 ， 选 择 “window” 一 “Android AVD Manager” 选 项 ， 将 
name、target、sdcard、skin 都 填 选 好 后 ， 单 击 “Create AVD” 即 可 。 
3) 在 CMD 窗口 中 进入 到 目录 下 ， 执 行 


emulator - avd avdname 
经 过 上 述 操作 后 ， 模 拟 器 就 运行 起 来 了 。 


注意 : 如 果 没 有 需要 的 JDK、Eclipse 和 Android SDK, 在 Linux 下 也 需要 分 别 下 载 它 们 ， 
只 是 在 下 载 时 选择 Linux 的 资源 即 可 ， 整 个 安装 顺序 和 Windows 下 大 同 小 异 。 


2.3.2 ”苹果 平台 下 的 搭建 过 程 
苹果 平台 下 搭建 Android 的 具体 过 程 和 Linux 下 的 类 似 ， 为 节省 本 书 篇 幅 ， 将 不 再 进行 
详细 介绍 。 只 是 在 下 载 Android SDK 的 时 候 ， 注 意 选择 Mac OS 版 本 的 资源 即 可 。 


2 本 相国 本 本 器 恒 


安装 和 搭建 一 个 开发 环境 ， 常 常会 遇 到 一 些 意 想不到 的 问题 ， 这 些 问题 可 能 是 粗心 造成 
的 ， 也 可 能 是 使 用 的 系统 环境 的 差异 造成 的 。 在 本 节 内 容 中 ， 将 简单 介绍 搭建 Android 开发 
环境 中 常见 问题 的 解决 方法 。 

1，Android 不 能 在 线 更 新 

在 安装 Android 后 ， 需 要 更 新 为 最 新 的 资源 和 配置 。 但 是 在 启动 Android 后 , 经 常会 不 能 
更 新 ， 弹 出 如 图 2-40 所 示 的 错误 提示 。 


Fetching https://dl-ssl. google. com/ android/repository/repository. xml 


Failed to fetch URL https://d1- 
ss1. go0gle. comf android/repository/repository. xml, reason: HTTFS 
SSL error. You might want to force download through HITP in the 
settings. 


图 2-40 不 能 更 新 


Android 默 认 的 在 线 更 新 地 址 是 https://dl-ssl.google.com/android/eclipse/， 但 是 经 常会 出 现 
背 误 。 如 果 此 地 址 不 能 更 新 ， 可 以 自行 设置 更 新 地 址 ， 修 改 为 http:/dl-ssl.google.comy/android/ 
repository/repository.xml。 具 体操 作 方法 如 下 。 
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1) 单 击 “Android SDK and AVD Manager” 窗 口 左 侧 的 “Available Packages” 选 项 ， 然 后 
单 击 下 面 的 “Add Site...” 按 钮 。 如 图 2-41 所 示 。 


入 hndroid SDk and AD Nanager 


Virtual Devices - - 
re 
ohttps: /dl-s:l om/ andr oid/repository/repository. xml 

OA Failed to fetch URL: HTTPS SSL error 

口 X No packages found 


人 


Description 
Source: https://dl-ssl. google. com/android/repository/repository. xml 


Add Site... | DETEte ore WY Display updates only Refresh|| Irstall Selected 


图 2-41 “Available Packages” 界 面 


2) 在 弹出 的 “Add Site URL” 对 话 框 中 输入 修改 后 的 地 址 http://dl-ssl.google.com/android/ 
repository/repository.xml， 如 图 2-42 所 示 。 


Add Site URL 


Flease enter the URL of the repository. xml for the new site: 


[httal Hr-ssl. googLe. com/ android/repository/repository. xml 
| 


图 2-42 “Available Packages” 界 面 
3) 单 击 “OK” 按 钮 ， 完 成 设置 。 经 过 上 述 操作 后 ， 就 能 够 使 用 更 新 功能 了 ， 如 图 2-43 
所 示 。 


Virtual Devices 
Installed Packages 


口 X Wo packages found 


8- 回廊 http: /7 四-ssl. google. com/android/repository/repository. xml 
由 - 团 意 Goo8le APIs by Google Ine., Mndroid API 3, revision 3 


SIK Source: https://d-ssl. go0gle. com/android/repository/repository. xml 


IY Display updates only Befresh | Install selected | 


Add Site... | 


图 2-43 “Available Packages” 界 面 
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一 直 显 示 “Project name must be specified 


第 2 章 _ 开 始 搭建 Android 开发 环境 “二 
2. Eclipse 中 新 建 Android 工程 时 ， 一 直 显 示 “Project name must be specified” 
”的 情况 如 


区 


2-44 所 示 。 


Project mame: 


FContents 


他 Create new project in workspace 
CT Create project from existing source 
VY Vse default location 


CT Create project from existing sample 


Louation FF /workspace ped | 


Source 
Source 
Source 
Source 
Source 


Project 
Project 
Project 
Project 
Project 


Sanples: |Flease select a target 
-Build Target 
OMmndroid 1.1 Android Dpen 
口 Mimaroid 1.5 Android Open 
口 ndiroid 1.6 Android Dpen 
口 inaroid 2.0 Android Dpen 
口 indroid 2.0.1 Android Dpen 
口 coogle AFIs Google Ine. 
口 Google MFIs Google Inc 
口 Google AFIs Google Ine. 
口 Google AFIs Google Ine. 
口 Google hrIs Google Ine. 
-Properties 


Application name: 


Package name; 


Min SDK Version: 


VT 
ES 
VY Create Activity: | 

EE 


图 2-44 


“Available Packages” 界 


< Back | Hext > | 区 | Cancel | 


总 


造成 上 述 问 题 的 原因 是 Android 没有 更 
1) 打开 
所 示 。 


V 


“Android SDK and AVD Manager”， 选 择 左 侧 的 “Installed Packages”， 如 


所 完成 ， 需 要 进行 完全 更 新 。 具 体 方法 如 下 。 


名 


2-45 


SDK Platform 
SDK Platform 
SDK Platform 
SDK Platform 
SDK Platform Android 2.0.1, 
注 Google APIs by Google Inc., 
沪 Google APIs by Google Ine., 
沪 Google AFIs by Google Inc., 
意 Google APIs by Google Ine., 
星 Google hPIs by Google Ine., 


Android 1.1, 


API 2, revision 1 
Android 1.5, API 3, revision 3 
Android 1.6, API 4, revision 2 
Android 2.0, API 5, revision 1 


API 6, revision 1 

Android API 4, revision 2 
Android API 5, 
Android API 6, 
Android API 6, 
Android API 6, 


revision 1 
revision 1 
revision 1 

1 


revision 


国 Vsb Driver package, revision 2 


Description 


Tools 
Revision 4 


Update A1... | 


图 2-45 


“Available Packages” 界 


Delete... Betresh | 
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2) 在 右 侧 列表 中 选择 “Android SDK Tools，revision4”， 在 弹出 
最 后 单 击 “Install Accepted” 按 钮 后 开始 安装 更 新 ， 如 图 2-46 所 示 。 


ad md 


窗口 中 选择 “Accept”， 


TPackage Description & License 一 一 一 一 
~ Google APIs by Google Inc., ... Package Description 习 


Android + Google AFIs 
Revision 3 
Requires SIK Platform MAndroid ArI 3 


Mrchive Description 

Archive for any OS 

Size: 33 MiB 

SHA1: 1f92abf3aTBbeBBaeB032257fcTB20acbd2b2e3a 


License 
This is the Android Software Development Kit 
License Agreement. 


1. Introduction 


1.1 The Android Software Development Kit i 

to in this License Aereement as the “SDK” am 
specifically including the Android system Er 
packaged APIs, and Google APIs add-ons) is 

licensed to you subject to the terms of this 

License Azreement. This License Aereement forms a 局 | 


© Reject © Aceept Ml 


[on dr rr es Install Accentea | caneel | 


图 2-46 “Available Packages” 界 面 


3. Eclipse 中 单 击 “Window” 一 “Preference”， 单 击 右 侧 的 “Android” 选 项 ，Target 
9 Target 选项 
常 来 说 ， 当 Android 开发 环境 搭建 完毕 后 ， 会 在 “Preference” 中 显示 存在 的 SDK 
人 如 图 2-47 所 示 。 


Android 达 Ye 


Android Preferences 


四 Mndroid 
由 -Amt SDK Location: [Fandroid-s dwindows Browse... | 
由 rs eR Note: The list of SDK Targets below is only reloaded once you hit“Apply or OK . 
由 -Java 
由 -RunyDebug Mndroid 1.1 Android Dpen Source Project 2 
由 .Tasks Android 1.5 Android Dpen Source Project 加 了 
由 .Teanm Android 1.6 Android Dpen Source Project 1.6 4 
由 .Usage Data Collector Android 2.0 Android Dpen Source Project 2.0 5 
Validation ad old 2. 0.1 no Dpen Source Project .了 6 
1.6 4 
四 -XL co 季 Ts Ceoga Tne. 2.0 5 
Gooele AFIs Gooele Inc. 2.0.1 6 
Google APIs Google Ine. 2.0.1 B 
Google AFIs Google Inc 2.0.1 6B 


Android + Google AFIs 


Restore Defaults | Apply 
7 
5 en | 


图 2-47 SDK Targets 列表 


但 是 往往 因为 各 种 原因 , 会 不 显示 SDK Targets 列表 , 在 界面 中 也 不 显示 , 并 输出 “Failed 
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to find an AVD compatible with target” 提 示 的 错误 问题 。 上 述 问题 的 原因 是 AVD 没有 创建 


二 


成 功 ， 如 果 出 现 上 述 问 题 ， 需 要 手动 安装 ， 当 然 前 提 是 Android 更 新 完毕 。 有 具体 解决 方法 


如 下 。 
1) 在 “运行 ”中 输入 “cmd” 打开 CMD 窗口 ， 如 


而 


2-48 所 示 。 


;Windows %P [版 本 5.1.2688] 
Pi 有 1985-2881 Microsoft Corp- 


C:\Documents and Settings Mdministrator> 


图 2-48 ”SDK Targets 列表 


2) 使 用 “android” 命 令 创建 一 个 Targets， 创 建 AVD。 按 照 “android create avd --name 


2” 日 . 碟 了 本 


<your_avd_name> --target <targetID> ”格式 创建 AVD， 其 中 “your avd_name” 是 需要 创建 的 
AVD 的 名 称 ， 如 图 2-49 所 示 。 


而 


C:\Documents and Settings dministrator>android create aud —-name aa -—-target 3 
Android 1.6 is a basic findroid platform. 
| 


图 2-49 CMD 界面 


在 如 图 2-41 所 示 的 窗口 中 创建 了 一 个 名 为 aa，targetID 为 3 的 AVD， 然 后 在 CMD 界面 
中 输入 “n” 即 完成 操作 ， 如 图 2-50 所 示 。 


[ey 


oN C:\WINDOWS\system32\cmd. exe 


ft Windows XP [版 本 5.1.26806] 
所 有 1985-2881 Microsoft Corp. 


C:\Documents and Settings Administrator>android create avd --name aa --target 3 


Android 1.6 is a basic findroid platform. 


Do vou wish to create a custom hardware profile [no Jn 
Created AUD "aa” based on findroid 1.6, with the following hardware config: 


hw.lcd.density=16@ 


C:\Documents and Settings Mdninistrator> 


图 2-50 ”CMD 界面 


.在 线 更 新 上 时， 提示“Failed to rename directory d\android-sdk-windows\tools to 


d: 人 -sdk-windows\tools\temp\ToolPackage. old01” 之 类 的 错误 


这 种 情况 通常 是 因为 杀毒 软件 或 防火 墙 对 这 个 文件 进行 了 访问 ， 建 议 读者 在 更 新 时 关闭 
面 加 29 


一 Android 开发 完全 实战 宝典 
杀毒 软件 和 防火 墙 。 如 果 还 不 行 可 进行 如 下 操作 。 
1) 在 android-sdk-windows 下 复制 tools 文件 夹 ， 得 到 “ 复 件 tools” 文 件 夹 
2) 运行 复 件 tools 文件 夹 里 的 android.bat， 在 出 现 的 “Android SDK and AVD Manager” 
上 执行 之 前 的 升级 步骤 即 可 成 功 。 记 得 在 此 步 之 前 关闭 已 打开 的 Android SDK.. Manager 
〈《 若 有 )。 
3) 删除 “ 复 件 tools”( 非 必要 )。 
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手机 开发 项 目 最 终 都 会 在 手机 屏幕 上 显示 出 来 ， 因 此 ， 手 机 屏幕 的 显示 效果 往往 会 影响 
到 消费 者 的 购买 欲望 。 本 章 将 详细 介绍 ， 通 过 Android 来 设置 用 户 人 机 界面 的 各 种 操作 方法 。 
希望 读者 能 够 举一反三 ， 为 进入 本 书后 面 知 识 的 学 习 打 下 基础 。 
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在 实际 应 用 中 , 经 常 需要 在 手机 屏幕 中 显示 一 段 简单 地 文字 ， 有 时 为 了 满足 现实 的 需求 ， 
还 要 对 这 行文 字 进 行 颜色 和 大 小 的 设置 修饰 ， 以 实现 我 们 需要 的 绚丽 效果 。 在 本 节 的 内 容 ! 
将 通过 一 个 具体 实例 来 讲解 在 屏幕 中 更 改 显示 文字 标签 的 方法 。 
3.1.1 设计 理念 

实际 上 ， 可 以 在 Layout 中 可 以 创建 一 个 对 象 ， 并 定义 strings.xml 中 的 字符 串 常数 ， 然 后 
通过 文本 组 件 TextView 中 的 setText 方法 ， 在 欲 加 载 程序 最 初 之 际 来 更 改 TextView 组 件 中 显 
示 的 文字 。 
3.1.2 具体 实现 


本 节 实 例 保存 在 “光盘 :3\wenzi\” 文 件 夹 内 ， 主 文件 src/wenzi/wenzi.java 是 此 项 目的 主 
要 文件 ， 用 于 调用 各 个 公用 文件 来 实现 具体 的 功能 。 具 体 代 码 如 下 。 


~ 


package irdc.wenzi; 

import android.app.Activity; 

import android.os.Bundle; 

/* 必 须 引 用 widget.TextView 接口 才能 在 程序 里 声明 TextView 对 象 */ 
import android.widget.TextView; 


public class wenzi extends Activity 
{ 
/* 必 须 引 用 widget.TextView 才能 在 程序 里 声明 TextView 对 象 */ 
private TextView mTextView0!1; 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
/# 载 入 main.xml Layout， 此 时 myTextView01:text 为 str 1 */ 
setContentView(R.layout.main); 


一 Android 开发 完全 实战 宝典 
翌 使 用 findViewBtId 函数 ， 利 用 了 D 找到 该 TextView 对 象 */ 
ImTextView01 = (TextView) findViewById(R.id.myTextView01); 
String str_2 = "欢迎 来 到 Android 世界 .."; 
mTextView01.setText(str 2); 


} 


TextView 对 象 里 的 setText 方法 支持 以 下 多 态 构 造 方法 。 


public final void setText(CharSequence text) 
public final void setText(int resid) 
public void setText(CharSequence text, TextView.BufferType type) 
public final void setText(int resid, TextView.BufferType type) 
public final void setText(char[ | text, int start, int len) 


在 此 ， 以 最 后 setText(char[] text int start, int len) 为 例 ， 第 一 个 参数 char 数组 作为 输出 依 
据 ， 第 二 个 参数 为 从 哪 一 个 元 素 索 引 开 始 选取 ， 第 三 个 参数 则 为 要 取出 多 少 个 元 素 ， 再 看 下 
面 的 代码 。 


char char 1[] = new char[S]; 


char 1[0] = 
char 1[1]='a 
char 1[2] = 
char 1[3]="1; 
char 1[4] = a 


mTextView01.setText(char 1,1,3); 


如 上 述 程序 所 示 ， 输 出 的 结果 是 “avi”， 因为 从 第 1 个 元 素 索 引 开 始 ， 共 取 3 个 
元 素 ; 最 后 则 要 提醒 读者 ，TextView.setTextView 不 支持 HTML TAG 的 输出 ， 所 以 即 
便 写成 这 样 : 


巧 


ImTextView01.setText("<a ref=\"http://shop. 
teac.idv.tw/MyBlogN"> 我 /a>"); 


实际 输出 时 ， 也 就 是 纯 文 本 而 已 ， 并 不 会 进行 HIML TAG 的 转换 。 但 若 撤 开 HTML TAG 之 
外 《如 "<" 开 头 的 标记 )， 在 TextView 对 象 里 加 上 了 android:autoLink="all" 语 句 ， 那 么 正文 中 
若 有 网 址 (http:/)， 是 可 以 被 显示 的 ， 以 下 这 个 范例 请 读者 实现 。 


<TextView xmlns:android="http://schemas.android.com/apk/res/android" 
android:layout width="fll parent" 
android:layout_ height="wrap_content" 
android:autoLink="all" 
android:text=" 请 访问 : http://xxx.com 
/> 


经 过 上 述 操作 设置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-1 所 示 。 
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为 了 满足 特殊 需求 ， 也 可 以 对 屏幕 画面 的 背景 颜色 进行 设置 。 本 将 通过 一 个 简单 的 实 


图 3-1 运行 效果 


例 ， 来 简要 介绍 更 改 屏 幕 窗 
3.2.1 设计 理念 


如 果 是 默认 的 运行 结果 , 窗口 的 底 


口 画面 背景 颜 


色 的 方法 。 


序 onCreate 创建 的 同时 ， 加 载 预先 定义 的 画面 颜色 。 


本 实例 程序 的 设计 方式 是 在 drawable 上 


会 是 深 黑色 , 这 是 SDK 默认 的 颜色 。 要 更 改 Activity 
类 里 的 窗口 底 色 有 许多 方法 ， 最 简单 的 方法 就 是 将 颜色 色 码 事先 定义 在 drawable 当中 ， 当 程 


指定 布局 页 面 的 背景 色 (BackGround) 为 白色 ， 


但 这 里 的 “白色 ”( 颜 色色 码 为 # 丁 FFFFFFF〉 预先 定义 在 drawable 当中 ， 当 程序 运行 时 ， 背 景 


就 会 变 为 白色 。 


这 是 指定 Activity Layout 背景 颜色 最 简单 的 方法 ， 在 范例 最 本 ， 则 将 示范 如 何 创建 色彩 


板 (color table )， 让 Android 手机 程序 可 以 像 使 月 


运行 阶段 。 


3.2.2 具体 实现 


本 节 实 例 保存 在 “光盘 :3\beijing\” 文 件 夹 内 ， 下 面 简 单 介 绍 主要 文件 


主 文件 src/beijing/beijing.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文 伯 


能 。 具 体 代 码 如 下 。 


package irdc.beijing; 


import android.app.Activity; 


import android.os.Bundle; 


public class beijing extends Activity 


{ 


日 “常数 ” 般 直 接 取 用 ， 并 反应 在 应 用 程序 的 


4 具体 含义 。 


F 来 实现 具体 的 功 


下 画面 33 


Android 开发 完全 实战 宝典 
/** Called when the activity is first created. */ 


(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


} 
} 


上 述 代码 继承 自 Activity 类 ， 并 在 重 写 方法 onCreate 之 初 ， 直 接 显示 R.layout.main 


Cmain.xml) 这 个 页 面 安排 的 布局 配置 。 
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实际 设计 中 最 常用 的 定义 颜色 常数 的 方法 ， 则 是 使 用 程序 控制 TextView 对 象 上 的 文字 或 
其 他 对 象 的 背景 色 ee 方法 )， ee 时 的 背景 色 亮 起 ， 或 当 
失去 焦 点 时 ， 又 恢复 成 原来 的 背景 色 等 。 本 节 将 通过 一 个 简单 的 实例 ， 介 绍 更 改 TextView 文 
字 颜 色 的 方法 。 

3.3.1 设计 理念 

本 实例 扩展 了 前 一 个 实例 的 实现 ， 预 先 在 布局 页 面 中 设计 好 两 行 TextView 文本 ， 并 在 运 
行 onCreate 同时 ， 通 过 两 种 程序 描述 方法 ， 实 时 更 改 原 来 布局 页 面 里 TextView 文本 的 背景 色 
以 及 文字 颜色 , 最 后 学 会 使 用 Android 默认 的 颜色 常数 (graphics.Color) 来 更 改 文字 的 前 景色 。 


3.3.2 ”具体 实现 
本 节 实 例 保 存在 “光盘 :3\yanse\” 文 件 夹 内 ， 下 面 简单 介 
1. 主 文件 
主 文件 src/yanse/yanse.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公 ) 
具体 代码 如 下 


package irdc.yanse; 


import irdc.yanse.R; 

import android.app.Activity; 

import android.content.res.Resources; 
import android.graphics.Color; 

import android.graphics.drawable.Drawable; 
import android.os.Bundle; 

import android.widget. TextView; 


public class yanse extends Activity 


{ 


private TextView mTextView0!1; 
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绍 主要 文件 的 具体 含义 。 


kt 体 的 功能 。 


文件 来 实现 


} 


通过 


创建 之 初 ， 以 findViewByld 方法 初始 化 为 布局 (main.xml) 旨 


private TextView mTextView02; 


(QOverride 


第 3 


public void onCreate(Bundle savedInstanceState) 


{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


二 


章 用 户 人 机 界面 设置 


mlextView01 = (TextView) findViewById(R.id.myTextView01); 


mTextView01.setText(" 使 


Resources resources = getBaseContext().getResources(); 


用 的 是 Drawable 背景 色 文本 。"); 


Drawable HippoDrawable = resources.getDrawable(R.drawable.white); 
mTextView01.setBackgroundDrawable(HippoDrawable); 


ImTextView02 = (TextView) findViewById(R.id.myTextView02); 
mTextView02.setTextColor(Color.MAGENTA); 


三 


上 述 代码 新 建 两 个 文本 类 成 员 变 量 


县 : 


mTextView01 与 mTextView02， 这 | 


个 变量 在 


四 


的 TextView 对 象 。 使 用 了 Resource 


类 以 及 Drawable 类 ， 分 别 创建 了 resources 对 象 以 及 HippoDrawable 对 象 ， 并 将 前 一 个 范例 中 


在 mTextView02 当中 ， 使 


更 改 文 字 的 前 景色 。 


2. 布局 文件 


在 页 面 布 局 上 ， 使 月 


<?xml version="1.0" encoding="utf-8"?> 


更 改 mTextView01 的 文字 底 纹 。 更 改 TextView 对 象 里 的 文字 ， 


所 创建 的 R.drawable.white 以 getDrawable 方法 加 载 , 最 后 则 调用 了 setBackgroundDrawable 来 


了 graphics.Color 里 的 颜色 名 


则 使 用 了 setText 方法 。 
数 。 接 着 ， 再 利用 setTextColor 


日 了 2 个 TextView 对 象 ， 关 键 在 于 下 面 的 代码 。 


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 


android:orientation="vertical" 
android:layout_width="fill parent" 
android:layout_ height="fill parent" 

> 

<TextView 
android:id="(@+Hid/myTextView01" 
android:layout width="fill parent" 
android:layout height="wrap_content" 
android:text="(@string/str_textviewO1" 
p> 

<TextView 
android:id="@+Hid/myTextView02" 
android:layout_width="fill parent" 


国 图 35 


”Android 开发 完全 实战 宝典 
android:layout_ height"wrap_ content" 


android:text="(@string/str_textview02" 
/> 


</LinearLayout> 


经 过 上 述 操 作 设 置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-2 所 示 。 


频 色 


图 3-2 运行 效果 


3.4 DTexMew0DD 


前 几 个 实例 讲解 了 文本 颜色 和 背景 颜色 的 设置 方法 ， 其 实 也 可 以 对 TextView 文字 进行 置 
换 处 理 。 本 节 将 通过 一 个 简单 实例 的 实现 过 程 ， 介 绍 置换 TextView 文字 的 方法 。 
3.4.1 设计 理念 

置换 TextView 文字 可 以 通过 CharSequence 数据 类 型 与 Resource ID 应 用 来 实现 。 从 一 开 
始 布局 文件 通过 Resource 初始 化 TextView 对 象 的 文字 ， 到 程序 中 动态 更 改 TextView 文字 ， 
但 要 如 何在 代码 里 取得 Resource 的 字符 串 呢 ? 在 Android 里 ， 确 实 有 一 些 方法 可 以 直接 以 
R.string.* 直 接 转换 了 D 为 Sting， 不 过 ， 这 样 的 数据 类 型 转换 是 非常 规 的， 甚至 是 不 妥 的 ， 正 
确 的 方法 是 利用 Context.getString 方法 来 取得 存放 在 global 里 的 Resource ID 。 以 下 这 个 范例 
将 示范 如 何在 程序 运行 时 (runtime)， 通 过 CharSequence 依据 Resource ID 取出 字符 串 ， 并 正 
确 更 改 TextView 的 文字 。 


3.4.2 具体 实现 

本 节 实 例 保 存在 “光盘 :\3\zhihuan\” 文 件 夹 和 内， 下面 简 

1. 主 文件 

主 文件 src/zhihuan/zhihuan.java 是 此 项 目的 主要 文件 , 调用 各 个 公用 文件 来 实现 具体 的 功 
能 。 此 处 主 代码 和 前 面 程 序 的 差异 主要 是 :在 更 改 mTextView02 对 象 的 文字 时 (setText 方法 )， 
合并 了 str 3 与 str 2 这 两 个 不 同 对 象 , 由 于 setText 方法 同时 支持 CharSequence 与 String 类 型 
的 参数 ， 所 以 在 此 示范 不 同 数据 类 型 的 字符 串 进行 同步 输出 。 具 体 代码 如 下 。 


站 


单 介绍 主要 文件 的 具体 含义 。 


package irdc.zhihuan; 


import irdc.zhihuan.R; 
import android.app.Activity; 
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import android.os.Bundle; 
import android.widget.TextView; 
public class zhihuan extends Activity 


{ 


private TextView mTextView02; 


(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


ImTextView02 = (TextView) findViewById(R.id.myTextView02); 
CharSequence str 2 = getString(R.string.str_2); 


String str 3 = "这 是 调用 的 Resource"; 
mTextView02.setText(str 3 十 Str 2); 


} 


2. 布局 文件 

在 页 面 布局 上 ， 在 main.xml 里 创建 了 两 个 TextView， 并 采 LinearLayout 的 方式 配置 ,一 
上 一 下 ， 在 运行 结果 中 id 为 myTextView01 的 TextView 并 没有 任何 文字 的 更 改 ， 维 持 一 开始 
的 str_1( 参 考 字 符 串 常数 里 的 文字 )， 但 在 程序 运行 后 ，id 为 myTextView02 的 TextView 进 
行 了 文字 的 实时 更 改 。 具 体 代码 如 下 。 


<?xml version="1.0" encoding="utf-8"?> 
<LinearLayout xmlns:android="http://schemas. 
android.com/apk/res/android" 
android:orientation="vertical" 
android:background="(@drawable/white" 
android:layout width="fill parent" 
android:layout height="fill parent" 
> 
<TextView 
android:id="(@+tid/myTextView01" 


= 


android:layout width="wrap content" 
android:layout height="wrap _content" 
android:text="(@string/str_1" 
android:layout x="30px" 
android:layout y='"S0px" 

> 

</TextView> 

<TextView 
android:id="(@+tid/myTextView02" 


= 从 


android:layout_ width="wrap content" 


android:layout_ height="wrap_content" 
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android:text="(@string/str 2" 
android:layout x="30px" 
android:layout y="70px" 
> 
</TextView> 


</LinearLayout> 


经 过 上 述 操作 设置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-3 所 示 。 
注意 : 虽然 在 values/strings.xml 里 定义 了 默认 的 字符 串 常数 ， 需 留意 若 遭 遇 如 “2??、“"”、 
”等 符号 时 ， 必 须 使 用 转 义 字符 “”， 例 如 ， 


4 
闭合 3:03 AM 


置换 处 理 模块 


图 3-3 运行 效果 


S50 


不 同 的 手机 有 不 同 的 分 辨 率 ， 分 辨 率 越 高 屏幕 越 清晰 。 在 当前 手机 市 场 中 ， 屏 幕 的 分 辩 
率 大 小 已 经 成 为 划分 手机 档次 的 一 个 标准 。 本 节 将 通过 一 个 简单 实例 的 实现 ， 介 绍 获取 手机 
分 辩 率 大 小 的 方法 。 


3.5.1 ”设计 理念 
在 开发 手机 应 用 程序 时 ， 除 了 对 底层 API 的 掌握 度 之 外 ， 最 重要 的 仍 是 对 屏幕 的 分 辩 率 
的 把 握 ， 因 各 家 手机 厂商 所 采用 的 屏幕 尺寸 不 同 ， 用 户 接口 呈现 及 布局 自然 也 各 异 。 

尽管 Android 可 设置 为 随 着 窗口 大 小 调整 缩放 比例 ， 但 即便 如 此 ， 手 机 程序 设计 人 员 还 
是 必须 知道 手机 屏幕 的 边界 ， 以 避免 缩放 造成 的 布局 变形 问题 。 这 个 范例 非常 简短 ， 只 需 几 
行程 序 即 可 取得 手机 的 分 辨 率 ， 其 中 的 关键 则 是 DisplayMetrics 类 的 应 用 。 


3.5.2 具体 实现 
本 节 实 例 保存 在 “光盘 :\3\fenbian\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 含义 。 
主 文件 src/fenbian/fenbian.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功 


能 。 具 体 代码 如 下 。 
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package irdc.fenbian; 


import irdc.fenbian.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.util.DisplayMetrics; 
import android.widget.TextView; 
public class fenbian extends Activity 
{ 


private TextView mTextView01; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/* 必须 引用 android.util.DisplayMetrics */ 
DisplayMetrics dm = new DisplayMetrics(); 
getWindowManager().getDefaultDisplay().get Metrics(dm); 


String strOpt = "当前 这 个 手机 屏幕 分 辨 率 是 : "+ 
dm.widthPixels + " x " + dm.heightPixels; 


mTextView01 = (TextView) fndViewById(R.id.myTextView01); 
mTextView01.setText(strOpt); 


} 
} 


在 android.util 下 有 一 个 名 为 DisplayMetrics 的 对 象 ， 此 对 象 记 录 了 一 些 常 用 的 信息 ， 包 
含 了 显示 人 信息、 大小、 维度、 字体 等 ; 在 使 用 时 ,请 读者 记得 引用 android.util.DisplayMetrics。 
其 中 DisplayMetrics 对 象 里 的 widthPixels 和 heightPixels 字段 是 整数 类 型 ,在 以 下 的 程序 当中 ， 
并 没有 对 其 作 字 符 串 类 型 的 转换 ， 因 为 字符 串 连 接 运 算 符 的 缘故 ， 所 以 输出 strOpt 为 字符 虽 
经 过 上 述 操作 设置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-4 所 示 。 


4 1 3 


Ud 


o 


手机 屏 丰 的 分 状 率 
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上 述 程序 一 开始 所 创建 的 DisplayMetrics 对 得 


造 时 )， 


(程序 中 的 dm)， 不 需要 传递 任何 参数 〈 构 


但 在 调用 getWindowManager() 方 法 之 后 ， 会 取得 现 有 的 Activity 组 件 中 的 Handler， 


此 时 ， 调 用 getDefaultDisplay 方法 将 取得 的 宽 、 高 维度 存放 于 DisplayMetrics 对 象 dm 中 ， 而 


取得 的 宽 、 高 维 


3.6 


后 


度 以 像素 为 单位 〈Pixel),“ 像 素 ” 所 指 的 是 “绝对 像素 ”而 非 “ 相 对 像素 ”。 


前 几 个 实例 讲解 了 某 个 单独 对 象 的 修饰 、 处 理 方法 。 如 果 一 个 个 地 指定 文字 的 大 小 、 颜 


色 很 浪费 时 间 ， 实 际 上 可 以 使 用 类 似 CSS 样式 的 方法 来 指定 颜色 、 大 小 。 本 节 将 通过 一 个 简 
单 实例 的 实现 过 程 ， 介 绍 样式 化 处 理 对 象 的 方法 。 
3.6.1 设计 理念 

在 Android 程序 开发 过 程 中 ， 也 可 以 通过 设置 样式 〈Style) 的 方式 ， 初 始 化 TextView 的 
文本 颜色 、 大 小 ;当然 这 个 范例 只 是 抛砖引玉 ， 在 Layout《〈 布 局 ) 当中 的 任何 对 象 都 可 以 用 
样式 化 的 方式 来 更 改 其 外 观 。 

在 以 下 的 范例 中 ， 将 创建 两 个 TextView 对 象 作为 对 比 ， 使 其 呈现 两 种 不 同 的 样式 差异 ， 


3.6.2 


具体 实现 


本 
1 . 


主 文件 src/yangshi/yangshi.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文 伯 


节 实 例 保 存在 “光盘 :3\yangshi\” 文 件 夹 内 


主 文件 


ZI 
CC 


k 体 实现 代码 如 下 。 


package irdc.yangshi; 


import irdc.yangshi.R; 
import android.app.Activity; 
import android.os.Bundle; 
public class yangshi extends Activity 
{ 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
} 
} 
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而 Style 的 写法 与 先前 介绍 到 的 颜色 常数 (color.xml) 相同 ， 同 样 是 定义 在 res/values 下 ， 但 
其 XML 定义 的 方式 不 同 。 来 看 下 面 的 这 个 实例 。 


， 下 面 简单 介绍 主要 文件 的 具体 合 义 。 


F 来 实现 具体 的 功 


上 述 代 码 可 知 主 程序 十 分 简单 ， 只 是 加 载 R.layout.main 定义 布局 内 容 而 已 ， 但 由 于 定 
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义 在 main.xml 里 的 语句 不 同 ， 上 自然 也 有 不 同 的 样 貌 。 

2. 布局 文件 

本 实例 的 布局 文件 由 main.xml 和 style.xml 共同 实现 。 其 中 文件 main.xml 是 纯 布 局 文件 ， 
设置 了 初始 化 TextView 时 ， 指 定 Style 属性 ， 使 其 应 用 style.xml 里 事先 定义 好 的 样式 。 

文件 style.xml 是 纯 修 饰 文件 ， 犹 如 CSS 样式 文件 一 样 ， 设 置 了 各 个 元 素 的 显示 样式 ， 并 
且 是 统一 指定 的 。 

style.xml 文件 是 这 个 实例 的 关键 所 在 ， 定 义 了 两 个 样式 名 称 ， 分 别 为 DavidStyleTextl 与 
DavidStyleText2; <style>TAG 里 以 <item> 描 述 的 属性 方式 , 与 先前 介绍 Drawable name 的 描述 
类 似 。 

经 过 上 述 操 作 设 置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-5 所 示 。 


欠 天 提 03:28 


样式 修饰 处 理 


作用 于 DavidStyleText1 


图 3-5 运行 效果 


注意 : style 与 color 的 XML 语法 相 类 似 ， 都 需要 先 声 明 XML 的 版 本 以 及 encoding 为 
utf-8， 但 其 内 的 resources 则 需要 以 stylename 作为 样式 名 称 ， 在 最 内 层 才 是 以 item 定义 样式 
的 范围 ， 其 具体 的 语法 如 下 。 


<style name=string [parent=string| > 
<item name=string>Hex value | string value | reference</item>+ 
</style> 


= 六 


前 几 个 实例 都 是 对 显示 样式 的 处 理 ， 从 本 节 开 始 讲解 和 动态 有 关 的 知识 。 按 钮 在 许多 
Windows 窗口 应 用 程序 中 ， 是 最 常见 到 的 组 件 〈Controls)， 此 组 件 也 常 在 网 页 设计 里 出 现 ， 
诸如 网 页 注册 窗 体 、 应 用 程序 里 等 。 本 节 将 通过 一 个 简单 实例 的 实现 ， 介 绍 Android 中 响应 
按钮 事件 的 方法 。 

3.7.1 设计 理念 

按钮 所 触发 的 事件 处 理 被 称 为 Event Handler〈 事 件 句 柄 )， 只 不 过 在 Android 当中 ， 按 钮 

事件 由 系统 的 Button.OnClickListener 所 控制 ， 熟 悉 Java 程序 设计 的 读者 对 OnXxxListener 应 


该 不 陌生 。 下 面 的 实例 将 展示 如 何在 Activity 组 件 中 布局 一 个 Button 〈 按 钮 )， 并 设计 这 个 按 
钮 的 事件 处 理 函数 ， 当 单 击 按钮 时 ， 更 改 TextView 里 的 文字 。 
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3.7.2 具体 实现 


本 节 实 例 保存 在 “光盘 :\3\anniu\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 含义 。 

主 文件 src/anniu/anniu.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功能 。 
在 此 文件 开始 ， 必 须 先 在 Layout 当中 加 入 一 个 Button 对 象 及 一 个 TextView 对 象 ， 找 不 到 这 
两 个 组 件 的 话 ， 系 统 会 无 法 运行 下 去 ， 在 开发 阶段 会 造成 编译 错误 。 具 体 实现 代码 如 下 。 


\NSN 
/ 
下 


package irdc.anniu; 


import irdc.anniu.R; 
import android.app.Activity; 
import android.os.Bundle; 
import android.view.View; 
import android.widget.Button; 
import android.widget.TextView; 
public class anniu extends Activity 
{ 
private Button mButton1; 
private TextView mTextView!l; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButtonl1 = (Button) findViewById(R.id.myButton!); 
ImTextViewl = (TextView) findViewById(R.id.myTextViewl); 
mButton1.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 

{ 

mTextViewl.setText("Hi, Everyone!!"); 

} 

六 
} 
} 


在 上 述 代码 中 ， 需 要 注意 onCreate 里 创建 的 Button.OnClickListener 事件 ， 这 也 是 触发 按钮 时 
会 运行 的 程序 段落 ;但 由 于 Eclipse 无 法 自动 加 载 默认 的 传递 参数 Cnew Button.OnClickListener())， 
所 以 ， 在 编写 程序 描述 时 ， 必 须 自 行 输入 新 创建 的 按钮 所 需 的 OnClickListener() 事件 。 

经 过 上 述 操作 设置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-6 所 示 。 当 
单 击 屏幕 中 的 按钮 后 ， 屏 幕 上 的 文本 将 会 发 生变 化 ， 由 原来 的 “anniu” 变 为 “Hi,Everyone”， 
42 国 国 


山中 


第 3 章 用 户 人 机 界面 设置 


如 图 3-7 所 示 。 


5554: aa 


响应 按钮 


图 3-6 运行 效果 
读者 应 该 注意 到 ， 上 述 代 码 中 只 有 一 个 按钮 ， 但 在 Activity 里 ， 其 实 可 以 布局 多 个 按钮 ， 
只 需要 在 Layout 里 多 配置 一 个 按钮 对 象 即 可 。 


[CT 


聊 应 按钮 
|, Everyone!! 


按 下 我 


图 3-7 运行 效果 


3 二 全 | 味 卫 性 加 轿 


在 上 网 冲浪 时 ， 人 们 经 常 去 不 同 的 页 面 ， 来 浏览 浩瀚 的 网 络 资 源 。 用 手机 上 网 时 ， 也 经 
常 需 要 访问 不 同 的 页 面 。 本 节 将 通过 一 个 简单 实例 的 实现 ， 介 绍 页 面 转换 处 理 的 方法 。 


3.8.1 设计 理念 

在 Android 应 用 中 ， 是 通过 setContentView 来 实现 页 面 的 转换 处 理 的 。 在 网 页 的 世界 里 ， 
想 要 在 两 个 网 页 间 做 转换 ， 只 要 利用 超 链 接 (HyperLink)〉 就 可 以 实现 ， 但 在 手机 的 世界 里 ， 
要 如 何 实现 手机 页 面 之 间 的 转换 呢 ?” 最 简单 的 方式 就 是 改变 Activity 的 Layout! 在 这 个 范例 
中 ， 将 布局 两 个 Layout， 分 别 为 Layoutl (main.xml) 与 Layout2 (mylayout.xml)， 默 认 载 入 
的 Layout 为 main.xml， 且 在 Layoutl 中 创建 一 个 按钮 ， 当 单 击 按钮 时 ， 显 示 第 二 个 Layout 
(mylayout.xml); 同样 地 ， 在 Layout2 里 也 设计 一 个 按钮 ， 当 单 击 第 二 个 Layout 的 按钮 之 后 ， 
则 显示 回 原来 的 Layout1， 现 在 就 来 示范 如 何在 两 个 页 面 之 间 互 相 切换 。 


3.8.2 ”具体 实现 
本 节 实 例 保存 在 “光盘 :\3\zhuanhuan\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 含义 。 
1. 主 文件 
主 文 件 src/zhuanhuan/zhuanhuan.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具 
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体 的 功能 。 有 具体 实现 代码 如 下 。 


package irdc.zhuanhuan; 


放 加 载 相 关 类 */ 

import irdc.zhuanhuan.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 

public class zhuanhuan extends Activity 

{ 
/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/*# 载 入 布局 文件 main.xml */ 


setContentView(R.layout.main); 


/* 通过 方法 fndViewById0 取 得 Button 对 象 ， 并 添加 onClickListener( 单 击 事件 ) 六 
Button bl = (Button) findViewById(R.id.buttonl ); 
bl.setOnClickListener(new Button.OnClickListener() 
{ 
public void onClick(View v) 


{ 
JumpToLayout2(); 


水 


/* method jumpToLayout2: 将 layout 由 main.xml 切换 成 mylayout.xml */ 


public void jumpToLayout2() 

{ 
/* 将 布局 改 成 mylayout.xml */ 
setContentView(R.layout.mylayout); 


/* 以 findViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
Button b2 = (Button) findViewById(R.id.button2); 
b2.setOnClickListener(new Button.OnClickListener() 
{ 

public void onClick(View v) 

{ 

JumpToLayout1(); 

} 

); 
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} 


/method jumpToLayout1: 将 1layout 由 mylayout.xml 切换 成 main.xml */ 
public void jumpToLayout1() 
{ 

/* 将 布局 改 成 main.xml */ 


setContentView(R.layout.main); 


上 以 findViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
Button bl = (Button) findViewById(R.id.buttonl ); 
bl.setOnClickListener(new Button.OnClickListener() 


{ 
public void onClick(View v) 


{ 
jumpToLayout2(); 
} 
); 
} 
} 


在 上 述 代 码 中 ， 预 加 载 的 布局 文件 是 main.xml， 屏 幕 上 显示 的 是 黑色 背景 的 “This is 
Layout 1!1”， 在 第 一 个 界面 上 的 按钮 被 单 击 的 同时 ， 改 变 Activity 的 布局 为 mylayout.xml， 屏 
幕 上 显示 变 为 白色 背景 的 “This is Layout 2!1”， 并 利用 按钮 单 击 时 ,调用 方法 的 不 同 ， 做 两 个 
不 同 布 局 间 的 切换 。 

2. 修饰 文件 

本 实例 的 布局 文件 由 main.xml 和 mylayout.xml 共同 实现 。 其 中 , 文件 main.xml 是 为 了 突 
出 显示 不 同 布局 界面 间 切 换 的 效果 ， 改 变 两 个 布局 的 背景 色 及 输出 文字 。 在 main.xml 中 定义 
其 缘 景 色 为 黑色 ， 输 出 文字 为 “This is Layout 1!!”。 
而 在 mylayoutxml 中 ， 定 义 了 其 背景 色 为 白色 ， 输 出 文字 为 “This is Layout 2!!”。 
经 过 上 述 操作 设置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-8 所 示 。 当 
单 击 屏幕 中 的 按钮 后 ， 屏 幕 上 的 文本 将 会 发 生变 化 ， 由 原来 的 “anniu” 变 为 “zhunahuan ”， 
如 图 3-9 所 示 。 


zhuanhuan 


zhuanhuan 


人 


图 3-8 运行 效果 


在 现实 应 用 中 ， 基 本 都 是 采用 改变 Activity Layout (界面 布 局 ) 来 实现 手机 页 面 转换 的 效 
果 ， 当 然 也 可 以 搭配 之 前 介绍 过 的 Style 与 Theme， 进 行 更 加 灵活 的 布局 配置 运用 。 例 如 ， 让 
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用 户 自行 决定 要 使 用 的 系统 样式 、 背 景 及 文字 颜色 等 ， 接 着 直接 应 用 来 改变 布局 。 


zhuanhuan 


zhuanhuan 


Go to Layout1 


图 3-9 转换 后 效果 
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在 上 一 个 实例 中 介绍 了 如 何 运 用 切换 Layout 布局 界面 的 方式 ， 进 行 手机 页 面 间 的 转换 。 
如 果 要 转换 的 页 面 并 不 单 只 是 背景 、 颜色 或 文字 等 内 容 ， 而 是 不 同 Activity 之 间 的 置换 ， 那 就 
不 是 仅仅 改变 Layout 就 能 完成 的 , 尤其 是 需要 传递 的 变量 不 像 网 页 可 以 通过 Cookie 或 Session 
来 实现 ， 在 程序 里 要 移交 主 控 权 到 另外 一 个 Activity， 光 靠 先 前 的 Layout 技巧 是 办 不 到 的 。 
本 节 将 通过 一 个 简单 实例 的 实现 过 程 ， 介 绍 调用 另 一 个 Activity 的 方法 。 


3.9.1 设计 理念 
在 Android 的 程序 设计 中 ， 可 在 主 程序 里 使 用 startActivity() 这 个 方法 来 调用 男 一 个 
Activity〈 主 程序 本 身 即 是 一 个 Activity), 但 其 中 的 关键 并 不 在 于 startActivity 这 个 方法 ,而 是 
Intent 这 个 特有 的 对 象 ，Intent 就 如 同 其 英文 字义 ， 是 “ 想 要 ”或 “意图 ”之 意 ， 在 主 Activity 
当中 ， 告 诉 程序 自己 是 什么 ， 并 想 要 前 往 哪 里 ， 这 就 是 Intent 对 象 所 处 理 的 事 了 。 本 范例 并 
没有 特别 的 Layout 布局 ， 而 是 直接 在 主 Activity (Activity1) 当中 部 署 一 个 按钮 ， 当 单 击 按钮 
的 同时 ， 告 诉 主 Activity 前 往 Activity2， 并 在 Activity2 里 创建 一 个 回 到 Activityl 的 按钮 ， 本 
范例 将 利用 此 简易 的 程序 描述 ， 示 范 如 何在 一 个 Activity 中 调用 男 一 个 Activity 的 方法 。 
3.9.2 具体 实现 

本 节 实 例 保存 在 “光盘 :3\zhihuan1\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 售 义 。 

1. 主 文件 


主 文件 src/zhihuan1/zihuan.java 和 src/zhihuan1/zihuan 1.java 是 此 项 目的 主要 文件 ， 调 用 
各 个 公用 文件 来 实现 有 具体 的 功能 。 文 件 zihuan.java 的 具体 实现 代码 如 下 。 


package irdc.zhihuan; 


旋 引入 相关 类 */ 

import irdc.zhihuan.R; 
import android.app.Activity; 
import android.os.Bundle; 
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import android.view.View; 
import android.widget.Button; 
import android.content. Intent; 
public class zhihuan extends Activity 
{ 
/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
/* 载 入 mylayout.xml Layout */ 
setContentView(R.layout.main); 


/* 以 findViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
Button bl = (Button) findViewBylId(R.id.button!1); 
bl.setOnClickListener(new Button.OnClickListener() 
{ 
public void onClick(View v) 
{ 
/# 新 建 一 个 Intent 对 象 ， 并 指定 要 启动 的 类 */ 
Intent intent = new Intent(); 


intent.setClass(zhihuan.this, zhihuan 1.class); 


/* 调用 一 个 新 的 Activity */ 
startActivity(intent); 

谍 关闭 原本 的 Activity */ 
zhihuan.this.finish(); 


}); 


在 上 述 代码 中 加 载 的 Layout 是 main.xml， 屏 幕 上 显示 的 是 黑色 背景 的 “Activity 1!1” 在 
按钮 被 单 击 时 调用 另 一 个 Activity (zhihuan 1)， 并 将 主 Activity 关闭 ， 接 着 将 主 控 权 交 给 下 
一 个 Activity， 即 Activity2。 

文件 zihuan_1.java 的 具体 实现 代码 如 下 。 


package irdc.zhihuan; 


Po SS 

import irdc.zhihuan.R; 

import android.app.Activity; 
import android.os.Bundle; 
import android.content. Intent; 
import android.view.View; 
import android.widget.Button; 
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public class zhihuan 1 extends Activity 


{ 
/** Called 


when the activity is first created. */ 


(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/# 载 入 布局 文件 main.xml */ 
setContentView(R.layout.mylayout); 


/* 以 fndViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
Button b2 = (Button) findViewById(R.id.button2); 
b2.setOnClickListener(new Button.OnClickListener() 


{ 


public void onClick(View v) 


{ 
Ws 


新 建 一 个 Intent 对 象 ， 并 指定 要 启动 的 类 */ 


Intent intent = new Intent(); 


intent.setClass(zhihuan 1.this, zhihuan.class); 
匀 调用 一 个 新 的 Activity */ 
startActivity(intent); 


/* 


关闭 原本 的 Activity */ 


zhihuan 1.this.finish(); 


j 


上 述 程 序 程序 是 第 二 个 Activity 的 主 程序 ， 其 加 载 的 布局 文件 为 mylayout.xml， 屏幕 


显示 的 是 白色 背景 


> 


的 “This is Activity 2!!”， 当 主 Activity (Activity1 ) 调 ) 


(Activity2) 后 ,同样 


为 Button 添加 onClickListener0， 使 Button 被 点 击 时 ， 重 六 


(zhihuan)， 并 将 Activity2 (zhihuan 1) 关闭 〈finishO) )。 


2. 修饰 文件 


本 实例 的 布局 文件 由 main.xml 和 mylayout.xml 共同 实现 ,其 
的 效果 ， 特 意 将 两 个 Layout 的 背景 及 输出 文字 有 所 区 别 。 在 main.xml 


显示 Layout 间 切 换 


定义 其 背景 色 为 黑色 ， 输 出 文字 为 “Activity 1!1!1”。 
在 文件 mylayout.xml 中 定义 其 背景 色 为 白色 ， 输 出 文字 为 “Activity 211”。 


3. 配置 文件 


六 


因为 在 该 实例 中 添加 了 一 个 Activity， 所 以 必须 在 AndroidManifest.xml ! 


定义 


activity， 并 给 予 名 称 name， 和 否则 程序 将 无 法 编译 运行 。 有 具体 实现 代码 如 下 所 示 。 


<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 


package="irdc.zhihuan" 
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这 个 Activity 
折 调 用 Activity1 


新 的 
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android:versionCode="1" 
android:version Name="1.0.0"> 
<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 
<activity android:name=".zhihuan" 
android:label="(@string/app_name"> 
<intent-filter> 
<action android:name="android.intent.action.MAIN"/> 
<category android:name="android.intent.category.LAUNCHER" /> 
</intent-filter> 
</activity> 
<activity android:name="zhihuan 1"></activity> 
</application> 


</manifest> 


经 过 上 述 操 作 设 置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-10 所 示 。 当 
单 击 屏 幕 中 的 按钮 后 ， 屏 蒂 上 的 文本 将 会 发 生变 化 。 如 图 3-11 所 示 。 


diaoyong 


Activity 1 


diaoyong 


Activity 2 


Go to Activity1 
| 


图 3-11 转换 后 效果 


系统 中 新 添加 Activity 时 ， 必 须 在 AndroidManifest.xml 里 定义 一 个 新 的 activity。 
<activity android:name="zhihuan 1"></activity> 


否则 ， 系 统 将 会 因为 找 不 到 Activity 而 发 生 编译 错误 。 
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在 上 节 的 实例 中 ， 介 绍 了 如 何在 Activity 中 调用 男 一 个 Activity， 但 是 在 现实 应 用 中 ， 经 
常 需要 在 不 同 的 Activity 之 间 传 递 数据 。 本 市 将 通过 一 个 简单 实例 ,介绍 在 不 同 Activity 之 间 
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传递 数据 的 方法 。 
3.10.1 设计 理念 

如 果 需 要 在 调用 男 外 一 个 Activity 的 同时 传递 数据 , 需要 利用 android.os.Bundle 对 象 封装 
数据 的 能 力 ， 将 欲 传 递 的 数据 或 参数 ， 通 过 Bundle 来 传递 不 同 Intent 之 间 的 数据 。 本 实例 将 
设计 一 个 简易 表单 类 型 ， 在 Activityl 中 收集 User 输入 的 数据 ， 在 离开 Activityl 的 同时 ， 将 
User 选择 的 结果 传递 至 下 一 个 Activity2， 以 一 个 简单 “身材 计算 器 ”来 演示 如 何 传递 数据 到 
下 一 个 Activity 里 。 


3.10.2 具体 实现 
本 节 实 例 保存 在 “光盘 :3\butong\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 含义 。 
1. 主 文件 
主 文件 src/butong/butong.java 和 src/butong/butong 1.java 是 此 项 目的 主要 文件 ， 调 用 各 个 
文件 来 实现 具体 的 功能 。 文 件 butong.java 的 具体 实现 代码 如 下 。 


Ns 
> 
Sp 


package irdc.butong; 


入 加 载 相关 类 * 

import irdc.butong.R; 

import android.app.Activity; 

import android.content. Intent; 

import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 

import android.widget.EditText; 

import android.widget.RadioButton; 

public class butong extends Activity 

{ 
/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 


super.onCreate(savedInstanceState); 
/# 载 入 布局 文件 main.xml */ 


setContentView(R.layout.main); 


/* 以 findViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
Button bl = (Button) findViewById(R.id.buttonl ); 
bl.setOnClickListener(new Button.OnClickListener() 
{ 
public void onClick(View v) 
{ 
放 取 得 输入 的 身高 */ 
EditText et = (EditText) findViewById(R.id.height); 
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double height=Double.parseDouble(et.getText().toString()); 
旋 取 得 选择 的 性 别 */ 


和 
? 


String Sex 
RadioButton rbl = (RadioButton) findViewByld(R.id.sex1); 
if(rb1.isChecked()) 
{ 
sex="M"; 
} 
else 
{ 
sex="F",; 
} 
/# 新 建 一 个 Intent 对 象 ， 并 指定 类 */ 
Intent intent = new Intent(); 


intent.setClass(butong.this,butong 1.class); 


/* 新 建 一 个 Bundle 对 象 ， 并 将 要 传递 的 数据 传 入 */ 
Bundle bundle = new Bundle(); 
bundle.putDouble("height",height); 
bundle.putString("sex",sex); 


人 # 将 Bundle 对 象 分 派 给 Intent*/ 
intent.putExtras(bundle); 


/* 调 用 Activity butong 1*/ 
startActivity(intent); 


文件 src/butong/butong 1.java 的 具体 代码 如 下 。 


package irdc.butong; 
入 加 载 相关 类 * 


import irdc.butong.R; 


import java.text.DecimalFormat; 


import java.text.NumberFormat; 


import android.app.Activity; 


import android.os.Bundle; 


import android.widget.TextView; 


public class butong 1 extends Activity 


{ 


/** Called when the activity is first created. */ 


， 定 义 了 “性 别 ” 选 项 的 单 选 按 钮 组 以 及 输入 身高 的 “EditText” 控 件 ，3 
运用 Intent 及 Bundle 对 象 ， 再 调用 Activity2 (butong 1) 时 ， 同 时 将 数据 传 入 。 
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(QOverride 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
访 加 载 布局 文件 main.xml */ 
setContentView(R.layout.myalyout); 


/* 取得 Intent 中 的 Bundle 对 象 */ 
Bundle bunde = this.getIntent().getExtras(); 


和 # 取得 Bundle 对 象 中 的 数据 */ 
String sex = bunde.getString("sex"); 
double height = bunde.getDouble("height"); 


庶 判断 性 别 */ 
String sexText=""; 
if(sex.equals("M")) 
{ 


sexText=" 男 性 "; 


} 


else 


{ 
sexText=" 女 性 "; 


入 取得 标准 体重 


String weight=this.getWeight(sex, height); 


此 设置 输出 文字 


/ 


TextView tvl= (TextView) fndViewById(R.id.textl); 
tvl.setText(" 你 是 一 位 "+sexTextt+"n 你 的 身高 是 " 
+height+" 厘 米 \n 你 的 标准 体重 是 "+weight+" 公 斤 "); 


旋 四 侈 五 入 的 方法 


这 


private String format(double num) 


{ 


NumberFormat formatter = new DecimalFormat("0.00"); 


String s=formatter.format(num); 


return s; 


} 


在 Activity2 (butong 1) 


需要 接收 来 


自 Activityl (butong) 传递 来 的 数据 。 在 Activityl 


是 以 Bundle 封装 对 象 ， 所 以 在 Activity2 也 是 以 Bundle 的 方式 解 开封 装 的 数据 ;程序 中 以 
getIntent().getExtras() 方法 取得 随 着 Bundle 对 象 传递 过 来 的 性 别 与 身高 , 经 过 计算 之 后 , 显示 
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在 屏幕 上 。 
2. 配置 文件 
于 本 范例 中 有 两 个 Activity， 所 以 文件 中 必须 有 两 个 Activity 的 声明 ， 否 则 系统 将 无 法 
运行 。 配 置 文件 AndroidManifestxml 的 具体 实现 代码 如 下 所 示 。 
<?xml version="1.0" encoding="utf-8"?> 


<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
package="irdc.butong" 


android:versionCode="1" 
android:version Name="1.0.0"> 
<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 
<activity android:name=".butong" 
android:label="(@string/app_name"> 
<intent-filter> 
<action android:name="android.intent.action.MAIN"/> 
<category android:name="android.intent.category.LAUNCHER" /> 
</intent-filter> 
</activity> 
<activity android:name="butong 1"></activity> 
</application> 
</manifest> 


经 过 上 述 操 作 设 置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-12 所 示 。 当 
输入 号 高 、 选 择 性 别 并 单 击 “ 计 算 ” 按 钮 后 ， 将 显示 标准 体重 。 如 网 3-13 所 示 。 


| 
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图 3-13 ”转换 后 效果 
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在 现实 应 用 ! 》 经 处 第 外 证 要 将 数据 返 回 到 前 一 个 Activity。 本 节 将 通过 个 简单 实例 ， 
将 数据 返回 到 前 一 个 Activity 的 方法 。 
3.11.1 设计 理念 

上 个 实例 描述 ， 将 数据 从 Activityl 传递 至 Activity2， 但 是 如 果 要 再 回 到 Activity1， 数 据 
是 否 再 封装 一 次 呢 ? 而 且 前 一 个 Activityl 早 就 被 程序 销毁 了 , 倘若 在 Activityl 最 后 以 finish() 
方法 结束 程序 ， 再 通过 Activity2 将 数据 采用 Bundle 的 方式 通过 新 打开 Activityl 传递 参数 ， 
这 样 的 做 法 虽然 也 可 以 恢复 User 输入 的 数据 ， 但 是 并 不 符合 我 们 的 期 待 。 

如 果 要 在 第 二 个 页 面 加 上 一 个 “ 回 上 页 ”的 按钮 ， 而 并 不 是 通过 模拟 器 中 的 回复 键 来 实 


现 ， 且 回 上 页 后 又 能 保留 之 前 输入 的 相关 信息 ， 那 么 就 必须 使 用 startActivityForResult0 来 唤 
起 一 个 Activity。 利 用 这 个 方法 ， 前 一 个 Activityl 便 会 有 一 个 等 待 第 二 个 Activity2 的 返回 ， 
而 返回 的 数据 就 可 以 实现 想 要 的 效果 。 
3.11.2 具体 实现 

本 节 实 例 保存 在 “光盘 :\3\qian\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 含义 。 

1. 主 文件 

主 文件 src/qian/qian.java 和 src/qian/qian 1.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 
来 实现 具体 的 功能 。 文 件 qian.java 的 具体 实现 代码 如 下 。 
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package irdc.qian; 


入 加 载 相关 类 */ 

import irdc.qian.R; 

import android.app.Activity; 

import android.content.Intent; 
import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 
import android.widget.EditText; 
import android.widget.RadioButton; 


public class qian extends Activity 
{ 
private EditText et; 
private RadioButton rbl; 
private RadioButton rb2; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
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super.onCreate(savedInstanceState); 


/* 载 入 布 


局 文件 main.xml */ 


setContentView(R.layout.main); 


/# 以 findViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
Button bl = (Button) findViewById(R.id.buttonl ); 
bl.setOnClickListener(new Button.OnClickListener() 


{ 


public void onClick(View v) 


{ 
/ 


e 
doubl 


* 


Ca 


取得 输入 的 身高 */ 
= (EditText) fndViewById(R.id.height); 


e height=Double.parseDouble(et.getText().toString()); 


/* 取 得 选择 的 性 别 */ 


String sex=""; 


rbl = (RadioButton) findViewBylId(R.id.sex1); 
rb2 = (RadioButton) fndViewById(R.id.sex2); 
iflrb1.isChecked()) 
{ 
sex="M"; 
} 
else 
{ 
sex="F"; 
} 


/* 新 建 一 个 Intent 对 象 ， 并 指定 类 */ 


Intent 


intent 


intent = new Intent(); 
.SetClass(qian.this,qian 1.class); 


人 # 新 建 一 个 Bundle 对 象 ， 并 将 要 传递 的 数据 传 入 */ 
Bundle bundle = new Bundle(); 


bundl 
bundl 


e.putDouble("height",height); 
e.putString("'sex",sex); 


/* 将 Bundle 对 象 分 派 给 Intent 对 象 */ 


intent 


.putExtras(bundle); 


/ 调 有 


有 Activity qian 1*/ 


startActivityForResult(intent,0); 


由 


二 


/* 履 盖 onActivityResult()*/ 
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(QOverride 
protected void onActivityResult(int requestCode, int resultCode, 
Intent data) 
{ 
switch (resultCode) 
{ 


case RESULT OK: 
/* 取得 来 自 Activity2 的 数据 ， 并 显示 于 画面 上 */ 
Bundle bunde = data.getExtras(); 
String sex = bunde.getString("sex"); 
double height = bunde.getDouble("height"); 


et.setText(""+height); 
if(sex.equals("M")) 
{ 
rbl.setChecked(true); 
} 
else 
{ 
rb2.setChecked(true); 
} 
break; 
default: 
break; 


} 


对 于 上 述 代码 , 在 Activityl 主 程序 中 , 将 调用 Activity 的 方法 更 改 成 startActivityForResult 
(intent,0)， 其 中 0 为 下 一 个 Activity 要 返回 值 的 依据 ， 可 指定 为 自行 定义 的 参考 标识 符 
(Identifier)。 程 序 履 盖 了 onActivityResult() 这 个 方法 ， 使 程序 在 收 到 result 后 ， 再 重新 加 载 写 
回 原本 输入 的 值 。 

对 于 文件 qian_1.java， 在 Activity2 的 主 程序 中 ， 设 计 当 按钮 被 单 击 时 ， 将 Bundle 对 象 与 
结果 返回 给 前 一 个 Activity1。 具 体 代 码 如 下 。 


package irdc.qian; 


入 加 载 相关 类 */ 
import irdc.qian.R; 


import java.text.DecimalFormat; 
import java.text.NumberFormat; 
import android.app.Activity; 
import android.content. Intent; 
import android.os.Bundle; 
import android.view.View; 
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import android.widget.Button; 
import android.widget. TextView; 


public class qian 1 extends Activity 
{ 
Bundle bunde; 
Intent intent; 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
/* 载 入 mylayout.xml Layout */ 
setContentView(R.layout.myalyout); 


从 取得 Intent 中 的 Bundle 对 象 */ 
intent=this.getIntent(); 
bunde = intent.getExtras(); 


人 # 取得 Bundle 对 象 中 的 数据 */ 
String sex = bunde.getString("sex"); 
double height = bunde.getDouble("height"); 


席 判断 性 别 六 
String sexText=""; 
if(sex.equals("M")) 


{ 

sexText=" 男 性 "; 
} 
else 
{ 

sexText=" 女 性 "; 
} 


旋 取得 标准 体重 */ 
String weight=this.getWeight(sex, height); 


上 # 设置 输出 文字 */ 

TextView tv1l=(TextView) findViewById(R.id.textl); 
tvl.setText(" 你 是 一 位 "+sexTexttm 你 的 身高 是 "+height+ 
"厘米 wn 你 的 标准 体重 是 "+weightt" 公 斤 "); 


访 以 fndViewById0 取 得 Button 对 象 ， 并 添加 onClickListener 单 击 监听 事 伯 
Button bl = (Button) findViewById(R.id.buttonl ); 
bl.setOnClickListener(new Button.OnClickListener() 


{ 


陛 


未 
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public void onClick(View v) 


{ 
放 返回 结果 回 上 一 个 activity */ 
qian 1.this.setResult(RESULT OK, intent); 


/# 结束 这 个 activity */ 
qian 1.this.finish(); 


了 


上/# 四 售 五 入 的 method */ 

private String format(double num) 

{ 
NumberFormat formatter = new DecimalFormat("0.00"); 
String s=formatter.format(num); 


return s; 


/# 以 findViewById0 取 得 Button 对 象 ， 并 添加 onClickListener */ 
private String getWeight(String Sex,double height) 
{ 


String weight=""; 
if(sex.equals("M")) 


{ 
weight=format((height-80)*0.7); 


} 


else 


{ 
weight=format((height-70)*0.6); 


} 


return weight; 


9 
2. 配置 文件 
由 于 本 范例 中 有 两 个 Activity， 所 以 文件 中 必须 有 两 个 Activity 的 声明 ， 否 则 系统 将 无 法 
运行 。 配 置 文件 AndroidManifest.xml 的 具体 实现 代码 如 下 。 


< ?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 


Ne 


package="irdc.qian" 
android:versionCode="1" 
android:versionName="1.0.0"> 
<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 


<activity android:name=".qian" 
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android:label="(@string/app_name"> 


<intent-filter> 


<action android:name="android.intent.action.MAIN"/> 
<category android:name="android.intent.category.LAUNCHER" /> 


</intent-filter> 


</activity> 


<activity android:name="qian 1"></activity> 


</application> 


</manifest> 


输入 身高 、 


F 述 操作 设置 ， 此 实例 的 主要 文件 编程 完毕 。 调 试 运行 后 的 效果 如 图 3-14 所 示 。 妆 


选择 性 


E 别 并 单 击 “ 计 算 ” 按 钮 后 ， 将 显示 标准 体重 。 如 图 3-15 所 示 。 
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图 3-15 ”转换 后 效果 
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在 上 述 实例 中 ， 为 了 在 回 到 上 一 页 时 ， 能 够 显示 之 前 所 输入 的 数据 ， 所 以 将 原本 传递 次 
Activity 的 Intent〈 里 面包 含 了 有 数据 的 Bundle 对 象 ) 再 重新 返回 给 主 Activity1。 如 果 要 在 次 
Activity2 中 返回 其 它 的 数据 ， 例 如 ， 经 过 计算 后 的 结果 、 数 据 ， 此 时 只 需 将 要 返回 的 数据 再 
放 入 Bundle 对 象 中 即 可 。 
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在 现实 应 用 中 ， 经 常 需 要 在 手机 屏幕 中 实现 交互 对 话 框 功能 。 本 市 将 通过 一 个 简单 实例 
的 实现 ， 介 绍 在 手机 中 实现 交互 对 话 框 功能 的 方法 。 


3.12.1 设计 理念 

在 网 络 中 读者 会 经 常 遇 到 “提示 ”“ 和 警告 ”或 “确认 ”之 类 的 对 话 框 ， 当 按 下 按钮 后 ， 
会 显示 出 一 些 提 示 信 息 。 在 手机 开发 中 ， 同 样 需要 开发 出 类 似 功 能 的 交互 对 话 框 。 

在 Android SDK 中 ， 有 很 多 的 窗口 ， 其 中 具有 交互 功能 的 是 AlertDialog 窗口 。 本 实例 的 
设计 理念 就 是 基于 AlertDialog 组 件 的 ， 仅 仅 通过 一 个 简单 的 “确认 ”按钮 来 作为 演示 ， 当 单 
击 这 个 按钮 后 ， 会 产生 AlertDialog 对 话 框 。 


3.12.2 具体 实现 
本 节 实例 保存 在 “光盘 :3vduihua\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 


主 文件 src/duihua/duihua.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功 
能 。 文 件 duihua.java 的 具体 实现 代码 如 下 。 


package irdc.duihua; 


import irdc.duihua.R; 

import android.app.Activity; 

import android.app.AlertDialog; 

import android.content.DialogInterface; 
import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 

public class duihua extends Activity 


{ 


private Button mButton1; 


/** Called when the activity is first created. */ 

@Override 

public void onCreate(Bundle savedInstanceState) 

{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
mButtonl = (Button) fndViewById(R.id.myButtonl ); 
mButtonl.setOnClickListener(new Button.OnClickListener() 


60O 看 而 


话 框 窗口 


如 图 


3-17 所 示 。 


在 上 述 代码 中 ， 设 计 了 按钮 事 人 


调试 运行 后 的 效果 如 图 3-16 所 示 。 当 单 避 
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Override 

public void onClick(View v) 
new AlertDialog.Builder(duihua.this) 
.SetTitle(R.string.app_about) 
.SetMessage(R.string.app_about msg) 
.setPositiveButton(R.string.str_ok, 

new DialogInterface.OnClickListener() 
{ 


public void onClick(DialogInterface dialoginterface, int 1) 


F， 当 按 下 按钮 和 时， 触发 onClick 单 击 事件 ， 以 new 方式 


打开 AlertDialog 对 象 AlertDialog.Builder 函数 ， 生 成 一 个 具有 Title、Message 和 确认 按钮 的 对 


[2 和 


击 这 里 ”按钮 后 , 将 弹出 一 个 对 话 框 。 
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在 现实 应 用 中 ， 经 常 需 要 置换 手机 屏 间 中 文字 的 颜色 。 本 节 将 通过 一 个 简单 实例 ， 介 
置换 手机 屏幕 中 文字 颜色 的 方法 。 


3.13.1 设计 理念 
在 前 面 的 实例 中 ， 读 者 了 解 了 多 种 与 TextView 文字 相关 的 处 理 方法 ， 并 且 也 了 解 了 与 按 
钮 相关 的 处 理 方法 。 本 实例 将 对 前 面 的 知识 进行 整合 ， 通 过 按钮 的 setOnClickListener 和 
onClick 方法 的 方式 ， 在 单 击 按钮 后 触发 setTextColor 方法 来 改变 文本 颜色 ， 并 且 创 建 一 个 字 
定义 的 颜色 数组 mColor， 当 单 击 按钮 时 ， 会 根据 这 个 颜色 数组 的 索引 值 的 变化 来 置换 
TextView 的 文字 颜色 。 
3.13.2 具体 实现 

本 节 实 例 保存 在 “光盘 :3\wenyan\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 

主 文件 src/wenyan/wenyan.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功 
能 。 文 件 wenyan.java 的 具体 实现 代码 如 下 。 


package irdc.wenyan; 


import irdc.wenyan.R; 

import android.app.Activity; 

/# 必 须 引 用 graphics.Color 才能 使 用 Color.* 的 对 象 */ 
import android.graphics.Color; 


import android.os.Bundle; 
import android.view.View; 


放 必 须 引 用 widget.Button 才能 声明 使 用 Button 对 象 */ 
import android.widget.Button; 


/必须 引 用 widget.TextView 才能 声明 使 用 TestView 对 象 */ 
import android.widget.TextView; 
public class wenyan extends Activity 
{ 
private Button mButton; 
private TextView mText; 
private int[] mColors; 
private int colornum; 


/** Called when the activity is first created. */ 
(QOverride 


public void onCreate(Bundle savedInstanceState) 


{ 
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super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 通 过 findViewById 构造 器 来 使 用 main.xml 与 string.xml 
中 button 与 textView 的 参数 */ 

mButton=(Button) fndViewById(R.id.mybutton); 

mText= (TextView) findViewByld(R.id.mytext); 


放声 明 并 构造 一 整数 array 来 存储 欲 使 用 的 文字 颜色 */ 
mColors = new int[| 

1 
Color.BLACK, Color.RED, Color.BLUE， 
Color.GREEN, Color.MAGENTA, Color.YELLOW 


治 


colornum=0; 


信使 用 setOnClickListener 让 按钮 聆听 事件 */ 
mButton.setOnClickListener(new View.OnClickListener() 
{ 
旋 使 用 onClick 让 用 户 点 下 按钮 来 驱动 变动 文字 颜色 */ 
public void onClick(View v) 
{ 
if (colornum < mColors.length) 


{ 


mText.setTextColor(mColors[colornum!]); 


colornumt+; 
} 
else 
colornum=0; 


通过 按钮 上 的 onClick 方法 实现 的 。 


在 上 述 代码 中 ， 使 用 了 setTextColor 方法 来 改变 文字 的 颜色 ， 驱 动 setTextColor 的 事 
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ES 


是 


调试 运行 后 的 效果 如 图 3-18 所 示 ; 当 单 击 “ 单 击 ” 按 钮 后 ， 文 本 将 会 改变 颜色 ， 如 
3-19 所 示 ; 再 次 单 击 “ 单 击 ” 按 钮 后 ， 文 本 还 会 改变 颜色 。 如 图 3-20 所 示 。 


[1 


和 55tw 


克 


图 3-18 ”运行 效果 和 3-19 ”改变 颜色 


者 5554:% Es 


图 3-20 ”改变 颜色 
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Se 


在 现实 应 用 中 ， 经 常 需 要 置换 手机 屏 磋 中 文字 的 字体 格式 。 本 习 将 通 
实现 ， 介 绍 置 换 手 机 屏幕 中 文字 字体 的 方法 。 


3.14.1 设计 理念 


一 个 简单 实例 的 


文字 的 字体 格式 主要 包括 文字 大 小 《Size〉 和 字体 (Font)。 改 变 字体 和 改变 颜色 的 原理 


一 样 ， 即 通过 按钮 对 象 的 Button.onClickListener 单 击 监 听 事 件 来 改变 TextView 控件 对 象 的 字 


体 大 小 和 字体 。 


在 TextView 对 象 中 有 许多 和 字体 有 关 的 方法 。 例如 ， 使 用 setTextSize 


使 ) 


-LL 


] setTypeface 方法 来 指定 字体 。 在 本 实例 中 将 涉及 两 个 按钮 ， 一 个 控制 文本 大 小 ， 一 个 控 


制 文本 字体 ， 并 通过 外 部 资源 assets， 引 用 外 部 字体 文件 True Type Font， 再 通过 i 类 


的 creatFromAsset 方法 ， 让 TextView 通过 SetTypeface 方法 来 改变 字体 。 


3.14.2 具体 实现 


本 节 实 例 保存 在 “光盘 \3\ziti\” 文 件 夹 内 ， 主 文件 src/zitin/zitijava 是 此 项 目 


通过 调用 各 个 公用 文件 来 实现 具体 的 功能 。 文 件 zitijava 的 具体 实现 代码 如 下 


package irdc.ziti; 


import irdc.ziti.R; 

import android.app.Activity; 

/* 必 须 引 用 graphics.Typeface 才能 使 用 creatFromAsset() 来 改变 字体 */ 
import android.graphics.Typeface; 


import android.os.Bundle; 

import android.view.View; 
import android.widget.Button; 
import android.widget. TextView; 


public class Ziti extends Activity 
{ 
/** Called when the activity is first created. */ 
private TextView mText; 
private Button sizeButton; 
private Button fontButton; 
(@Override 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
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mText=(TextView)findViewByld(R.id.mytextview); 
sizeButton=(Button) findViewBylId(R.id.sizebutton); 
fontButton=(Button) findViewById(R.id.fontbutton); 
/# 设 置 onClickListener 与 按钮 对 象 连接 */ 
sizeButton.setOnClickListener(new View.OnClickListener() 
{ 
public void onClick(View v) 
{ 
放 使 用 setTextSize0 来 改变 字体 大 小 */ 
mText.setTextSize(20); 


} 


); 
fontButton.setOnClickListener(new View.OnClickListener() 
1 
public void onClick(View v) 
{ 
任 必 须 事先 在 assets 底下 创建 一 fonts 文件 夹 
* 并 放 入 要 使 用 的 字体 文件 (.ttf) 
* 并 提供 相对 路 径 给 creatFromAsset0 来 创建 Typeface 对 象 */ 
mText.setTypeface 


(Typeface.createFromAsset(getAssets(), 
"fonts/HandmadeTypewriter.ttf"')); 


调试 运行 后 的 效果 如 图 3-21 所 示 ; 当 单 击 “ 转 换 字体 ”按钮 和 “ 变 大 ”按钮 后 ， 文 本 样 
式 将 会 改变 。 如 图 3-22 所 示 。 
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图 3-21 运行 效果 


在 本 实例 中 ， 使 用 了 类 Typeface， 并 使 用 了 外 部 字体 文件 来 改变 文字 的 字体 。 但 是 在 当 


前 的 Android 中 , 缺乏 对 字体 的 支持 , 即使 使 用 了 它 不 支持 的 字体 , 也 不 会 报错 , 而 是 以 Droid 
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Type 来 替换 。 所 以 当 使 用 外 部 字体 但 字体 没有 变换 时 ， 通 常 是 因为 这 个 字体 在 Android 中 不 
被 支持 。 
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在 很 多 手机 应 用 中 ， 可 以 拖 动 一 幅 图 片 ， 这 样 的 效果 很 吸引 用 户 的 眼球 。 本 市 将 通过 一 
个 简单 实例 ， 介 绍 在 手机 屏幕 中 拖 动 图 片 特效 的 方法 。 


3.15.1 设计 理念 

在 Android 中 ， 拖 动 图 片 特效 可 以 通过 Android.content.Context、Android.widget.BaseAdapter 和 
Android.widgetImageView 等 来 实现 , 这 些 通 常 被 应 用 到 相册 和 图 片 类 型 选择 器 上 。 要 理解 实例 ， 
需要 了 解 Context 子 类 和 widget 里 的 BaseAdapter 类 。 在 Activity 中 ，Context 犹如 画布 ， 随 时 
会 被 处 理 覆 盖 。Context 是 作为 Android.content 的 子 类 。 在 本 实例 中 ， 在 Layout 中 布局 一 个 
Gallery〈 切 换 组 件 )， 再 通过 widgetBaseAdapter 作为 容器 来 存放 Gallery 所 需要 的 图 片 ， 为 了 
快速 掌握 Gallery 的 使 用 方法 ， 在 实例 中 使 用 了 Android 的 Icon 图 标 。 


3.15.2 具体 实现 

本 节 实 例 保 存在 “ 光 可 :3\texiao\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 

主 文件 src/texiao/texiao.java 是 此 项 目的 主要 文件 , 调用 各 个 公用 文件 来 实现 具体 的 功能 。 
文件 texiao.java 的 具体 实现 代码 如 下 。 


package irdc.texiao; 


import irdc.texiao.R; 

import android.app.Activity; 
import android.graphics.Color; 
import android.os.Bundle; 
import android.widget. TextView; 


/* 欲 在 Layout 里 使 用 Gallery widget， 必 须 引 用 这 些 模块 */ 
import android.content.Context; 


import android.widget.Gallery; 
import android.view.View; 
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import android.view.ViewGroup; 
import android.widget.BaseAdapter; 
import android.widget.ImageView; 


public class texiao extends Activity 


{ 


private TextView mTextView01; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mTextView01 = (TextView) findViewById(R.id.myTextView01); 
mTextView01.setText(getString(R.string.str txt1)); 
mTextView01.setTextColor(Color.BLUE); 


((Gallery) findViewById(R.id.myGallery1)) 
.setAdapter(new ImageAdapter(this)); 


public class ImageAdapter extends BaseAdapter 


{ 
/* 类 成 员 myContext 为 Context 父 类 *%/ 
private Context myContext; 
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/* 使 用 android.R.drawable 里 的 图 片 作为 图 库 来 源 ， 类 型 为 整数 数组 */ 


private int[] myImagelds = 
{ 
android.R.drawable.btn minus, 
android.R.drawable.btn_radio, 
android.R.drawable.ic lock idle low battery, 
androld.R.drawable.ic _ menu camera 
上 
上 # 构造 器 上 只 有 一 个 参数 ， 即 要 存储 的 Context */ 
public ImageAdapter(Context c) { this.myContext = c; } 


具 返回 所 有 已 定义 的 图 片 总 数量 党 
public int getCount() { return this.myImagelds.length; } 


旋 利用 getttem 方法 ， 取 得 目前 容器 中 图 像 的 数组 ID */ 
public Object getItem(int position) { return position; } 


public long getItemld(int position) { return position; } 
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/* 取得 目前 欲 显示 的 图 像 View， 传 入 数组 ID 值 使 之 读 取 与 成 像 */ 


public View getView(int position, View convertView, 


ViewGroup parent) 


{ 
/* 创建 一 个 ImageView 对 象 */ 
ImageView i= new ImageView(this.myContext); 


i.setImageResource(this.myImagelds[position]); 
i.setScaleType(ImageView.ScaleType.FIT XY); 


设置 这 个 ImageView 对 象 的 宽 高 ， 单 位 为 dip */ 
i.setLayoutParams(new Gallery.LayoutParams(120, 120)); 
return 1; 


} 


/依据 距离 中 央 的 位 移 量 利用 getScale 返回 views 的 大 小 (0.0f to 1.0f)*/ 
public float getScale(boolean focused, int offset) 
{ 
/* Formula: 1 / (2 人 offset) */ 
return Math.max(0,1.0f/(float)Math.pow(2,Math.abs(offset))); 
} 
} 
} 


在 上 述 代码 中 ， 创 建 了 一 个 继承 于 类 BaseAdapte 的 ImageAdapte 方法 。 此 ImageAdapte 
方法 的 功能 是 暂时 保存 要 显示 的 图 片 ， 作 为 Gallery 组 件 图 片 的 引用 。 

调试 运行 后 的 效果 如 图 3-23 所 示 ; 当 单 击 某 个 图 片 后 ， 会 以 特效 滚动 的 样式 显示 。 
如 图 3-24 所 示 。 


上 


68 而 画 再 


第 3 章 _ 用 户 人 机 界面 设置 “二 
上 述 实例 是 使 用 了 Android 的 内 置 图 标 , 另外 也 可 以 从 外 部 导入 图 片 ,只 需 在 res\drawable 
文件 夹 下 导入 图 形 文件 即 可 。 这 些 图 形 文件 会 在 部 署 阶段 同 应 用 程序 一 起 打包 成 .apk。 
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计算 器 是 手机 中 的 最 常用 程序 之 一 ， 通 过 计算 器 ， 可 以 方便 人 们 对 日 党 计算 的 处 理 。 本 
节 将 通过 一 个 简单 实例 ， 介 绍 在 手机 上 制作 计算 器 的 方法 。 
3.16.1 设计 理念 
在 Android 中 ， 我 们 可 以 使 用 Button 组 件 作为 计算 器 中 的 数字 按键 和 运算 符号 按键 ， 
将 计算 结果 显示 在 TextView 当中 。 每 次 单 击 一 个 Button 组 件 ， 都 会 响应 执行 一 个 处 理事 件 ， 
此 功能 是 通过 onClick(View Vv) 实 现 的 ， 为 每 个 Button 按钮 设置 了 监听 事件 方法 
setOnClickListener()。 


O 


IT 


3.16.2 具体 实现 

本 节 实 例 保存 在 “光盘 :\3Vjisuanqi\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 

主 文件 src/jisuanqi/jisuanqi.java 是 此 项 目的 主要 文件 , 调用 各 个 公用 文件 来 实现 具体 的 功 
能 。 文 件 jisuanqi.java 的 具体 实现 代码 如 下 。 


package irdc.jisuanqi; 


. 


public class jisuanqi extends Activity { 
/** Called when the activity is first created. */ 


public Button mButton2; 
public Button mButton3; 
public Button mButton4; 
public Button mButton5; 
public EditText mEditTextl; 
public EditText mEditText2; 
public TextView mTextView2; 
public TextView mTextView4; 


(@Override 
public void onCreate(Bundle savedInstanceState) 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mTlextView2 = (TextView) findViewByld(R.id.mTextView2); 
mTlextView4 = (TextView) findViewByld(R.id.mTextView4); 
mButton2 = (Button) fndViewById(R.id.mButton2 ); 
mButton3 = (Button) findViewById(R.id.mButton3); 
mButton4 = (Button) fndViewById(R.id.mButton4); 
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mButton5 = (Button) findViewById(R.id.mButton5); 


mEditTextl = (EditText) fndViewById(R.id.mTextl); 
mEditText2 = (EditText) fndViewById(R.id.mText2); 


mButton2.setOnClickListener(new Button.OnClickListener() 


QOverride 
public void onClick(View v) 
{ 

mTextView2.setText("+"); 

String strRet = Integer.toString( Integer.parseInt(mEditText1. 
getText().toString())+ Integer.parseInt 
(mEditText2.getText().toString()) ); 

mTextView4.setText(strRet); 


)); 
mButton3.setOnClickListener(new Button.OnClickListener() 


@Override 
public void onClick(View V) 
{ 

mTextView2.setText("-"); 

String strRet = Integer.toString( Integer.parseInt(mEditText1. 
getText().toString())- Integer.parseInt 
(mEditText2.getText().toString()) ); 

mTextView4.setText(strRet); 


D); 
mButton4.setOnClickListener(new Button.OnClickListener() 


@Override 
public void onClick(View V) 

mTextView2.setText("*"); 

String strRet = Integer.toString( Integer.parseInt(mEditText1. 
getText().toString())* Integer.parseInt 
(mEditText2.getText().toString()) ); 

mTextView4.setText(strRet); 
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mButton$.setOnClickListener(new Button.OnClickListener() 
{ 
Override 
public void onClick(View v) 
{ 

mTlextView2.setText("/"); 

String strRet = Integer.toString( Integer.parseInt(mEditText1. 
getText().toString())/ Integer.parseInt 
(mEditText2.getText().toString()) ); 

mTextView4.setText(strRet); 


在 上 述 代 码 中 ， 特 别 要 注意 对 象 的 名 称 ， 一 定 要 创建 按钮 对 象 。 因 为 需要 用 到 届 辑 运算 ， 
所 以 如 果 要 输出 数值 ， 就 需要 以 Java IntegertoString0 或 FloattoString() 来 处 理 这 些 数值 。 

调试 运行 后 的 效果 如 图 3-25 所 示 ; 当 输 入 数值 ， 选 择 一 个 算法 后 ， 将 会 计算 出 对 应 的 结 
果 。 如 图 3-26 所 示 。 


图 3-26 计算 结果 


上 述 实 例 只 能 计算 整数 ， 不 能 计算 小 数 类 型 。 可 以 对 其 进行 修改 ， 将 Integer 改 为 Float 
数据 类 型 ， 这 样 就 可 以 计算 小 数 了 。 
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关于 信息 的 作用 是 说 明 当 前 软件 或 硬件 的 基本 信息 ， 常 见于 计算 机 领域 。 例 如 ， 网 站 
中 的 “关于 我 们 ” 本 节 将 通过 一 个 简单 实例 的 实现 , 介绍 在 手机 中 实现 设置 About (关于 ) 
信息 的 方法 。 


3.17.1 设计 理念 

在 Android 中 ， 手 机 接口 是 Menu Shotcut， 即 所 谓 的 Menu Key。 本 实例 将 讲解 Android 
Menu Key 的 设计 方法 ， 演 示 “ 关 于 ”对 话 框 、“ 离 开 ” 对 话 框 的 语法 。 在 程序 中 除了 默认 履 
盖 的 onCreate 外 ， 还 需要 建立 2 个 类 函数 onCreateMenu0 和 onOptionItemSelected()。 其 中 ， 
前 者 是 创建 Menu 菜单 项 目 ， 后 者 则 是 处 理 菜 单 被 选择 运行 后 的 事件 。 在 实例 最 后 当 用 户 单 
击 “ 关 于 ”菜单 后 ， 会 弹出 AlertDialog， 显 示 出 “关于 ”的 信息 。 


3.17.2 具体 实现 

本 节 实 例 保 存在 “光盘 :\3\guanyu\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 

主 文件 src/guanyu/guanyu.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功 
能 。 文 件 guanyu.java 的 具体 实现 代码 如 下 。 


package irdc.guanyu; 
public class guanyu extends Activity 
{ 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


} 


public boolean onCreateOptionsMenu(Menu menu) 
{ 
menu.add(0, 0, 0, R.string.app_about); 
menu.add(0, 1, 1, R.string.str exit); 
return super.onCreateOptionsMenu(menu); 


} 


public boolean onOptionsItemSelected(Menultem item) 


{ 


super.onOptionsItemSelected(item); 
switch(item.getltemld()) 


{ 


72 故国 


case 0: 
openOptionsDialog(); 
break; 

case 1: 
finish(); 
break; 

} 
return true; 


} 


private void openOptionsDialog() 
{ 
new AlertDialog.Builder(this) 
.setTitle(R.string.app_ about) 
.setMessage(R.string.app about msg) 
.setPositiveButton(R.string.str_ok, 
new DialogInterface.OnClickListener() 
{ 
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public void onClick(DialogInterface dialoginterface, int 1) 


{ 
} 
} 
) 
.Show(); 
} 
} 


在 
菜单 项 ， 然 后 


调试 运行 后 的 效果 如 图 3-27 所 示 ; 当 单 击 
当 单 击 “ 关 于 ” 


于 


CE 


About 关 于 


3-27 运行 结果 


上 述 代 码 中 ， 创 建 了 一 个 onCreateOptionsMenu(Menu menu) 类 函数 ， 用 于 添加 Menu 
再 利用 onOptionItemSelected0 选 择 事件 获取 荣 单 选择 项 目 ， 处 理 对 应 的 事件 ， 
即 getItemId0=0 是 “关于 ” getttemIdO=1 是 “ 离 玫 


F 2 


o 


“Menu” 沫 单 后 会 弹出 两 个 子 染 单 ， 如 图 3-28 所 示 ; 
菜单 后 会 弹出 一 个 对 话 框 ， 这 个 对 话 框 就 是 “关于 ”的 说 明 信 息 。 如 图 


3-29 所 示 。 
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图 3-29 “关于 ”对 话 框 
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“程序 加 载 中 ” 常见 于 各 种 应 用 软件 ， 表 示 程 序 还 没有 加 载 完 成 ， 请 用 户 慢 慢 等 待 。 例 如 ， 
经 常 显示 “程序 正在 加 载 中 ， 请 稍 后 ..…”。 本 节 将 通过 一 个 简单 实例 ,介绍 在 手机 中 实现 “ 程 
序 加 载 中 ， 请 稍 后 .…” 效 果 的 方法 。 
3.18.1 ”设计 理念 

我 们 经 常 在 Windows 程序 中 看 到 “程序 正在 加 载 中 ”的 提示 。 在 Android 中 ， 此 功能 是 
通过 Progress Dialog 类 来 运行 ， 此 类 被 封装 在 Android.app.ProgressDialog 类 里 ， 但 需要 注意 
的 是 Android 中 的 Progress Dialog 类 必须 在 后 台 程 序 运行 完毕 前 , 使 用 dismiss0 方 法 来 关闭 取 
得 焦点 的 对 话 框 ， 否 则 程序 将 会 陷入 无 法 终止 的 无 穷 循环 中 ;或 者 在 线程 里 不 可 有 任何 更 改 
Context 或 parent View 的 任何 状态 、 文字 输出 等 事件 。 因为 线程 里 的 Context 或 View 并 不 属 
于 parent， 两 者 之 间 没 有 关联 。 在 本 实例 中 ， 将 以 线程 来 模拟 后 台 程 序 的 运行 ， 再 通过 线程 
运行 完毕 时 ， 关闭 这 个 加 载 中 的 动画 对 话 框 。 在 实例 中 ,将 设计 一 个 按钮 ， 单 击 按钮 后 开始 
线程 的 周期 ， 在 运行 过 程 中 显示 ProgressDialog 对 话 框 ， 当 线程 运行 完毕 后 ， 结 束 
ProgressDialog 对 话 框 。 


3.18.2 具体 实现 


本 节 实 例 保 存在 “光盘 :3\dengdai” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 
主 文 件 sre/dengdai /dengdai.java 是 此 项 目的 主要 文件 ， 调用 各 个 公用 文件 来 实现 具体 的 
功能 。 文 件 guanyu.java 的 具体 实现 代码 如 下 。 


package irdc.dengdai; 
public class dengdai extends Activity 


{ 


private Button mButton1; 
private TextView mTextView!; 
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public ProgressDialog myDialog = null; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButtonl =(Button) findViewById(R.id.myButton!); 
ImTextViewl = (TextView) findViewById(R.id.myTextViewl); 
mButtonl.setOnClickListener(myShowProgressBar); 


Button.OnClickListener myShowProgressBar = 
new Button.OnClickListener() 
{ 
public void onClick(View arg0) 
{ 
final CharSequence strDialogTitle = 
getString(R.string.str_ dialog title); 
final CharSequence strDialogBody = 
getString(R.string.str dialog body); 


/* 显示 Progress 对 话 框 */ 
myDialog = ProgressDialog.show 
( 
dengdai.this, 
strDialogTitle, 
strDialogBody, 
true 


); 
mTextViewl.setText(strDialogBody); 


new Thread() 
{ 
public void run() 
{ 
try 
{ 
入 在 这 里 写 上 要 运行 的 程序 片段 */ 
/* 为 了 明显 看 见效 果 ， 以 暂停 3 秒 作 为 示范 *% 
Sleep(3000); 
} 


catch (Exception e) 
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{ 
e.printStackTrace(); 

} 

finally 

{ 
入 印 载 所 创建 的 myDialog 对 象 */ 
myDialog.dismiss(); 

上 

} 
}.start(); /* 开始 运行 运行 线程 */ 
} /End: public void onClick(View arg0)*/ 


中 
} 
在 上 述 代 码 中 ， 创 建 了 一 个 按钮 ， 当 单 击 按钮 后 ， 会 触发 myShowProgressBar 事件 ， 在 
事件 中 更 改 了 TextView 里 的 文字 ， 并 将 焦点 传递 给 前 台 的 ProgressDialog.show 方法 。 当 运行 


3s 后 ， 再 将 焦点 传递 返回 给 原来 的 Activity。 
调试 运行 后 的 效果 如 图 3-30 所 示 ; 当 单 击 “ 按 下 后 ”按钮 后 会 弹出 “请 稍 后 ”提示 信息 ， 


如 图 3-31 所 示 。 
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图 3-31 显示 提示 信息 
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上 述 功能 是 基于 Android.app.ProgressDialog 类 实现 的 , Android.app.ProgressDialog 中 包括 


了 几 种 重要 的 构造 方法 ， 有 具体 信息 如 表 3-1 所 示 。 
表 3-1 Android.app.ProgressDialog 类 的 构造 函数 
方 法 格式 
static show(Contextcontext,CharSequencetitle,CharSequencemessage) 


show(Contextcontext,CharSequencetitle,CharSequencemessage, 
booleanindeterminate) 
show(Contextcontext,CharSequencetitle,CharSequencemessage, 
booleanindeterminate,booleancancelable) 
show(Contextcontext,CharSequencetitle,CharSequencemessage, 
static booleanindeterminate,booleancancelable, 
DialogInterface.OnCancelListenercancelListener) 


static 


static 


有 关 Android.app.ProgressDialog 类 更 加 详细 的 信息 ， 请 读者 参阅 下 面 的 网 址 。 


http:/www.chinaup.org/docs/reference/android/app/ProgressDialog.html 
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对 话 框 是 网 络 中 常见 的 窗口 ， 手 机 软件 中 也 必 不 可 少 。 前 面 内 容 中 已 经 介绍 了 对 话 框 的 
生成 方法 ， 本 节 将 进一步 介绍 一 种 更 为 复杂 的 对 话 框 一 内 容 可 选 的 对 话 杠 。 本 节 将 通过 
个 简单 实例 的 实现 ， 介 绍 实现 可 选择 对 话 框 的 方法 。 


3.19.1 设计 理念 
在 前 面 的 内 容 中 ,介绍 了 AlertDialog.Builder 对 话 框 ,其 实在 这 个 对 话 框 内 还 可 以 包含 对 
话 窗 口 ， 即 含有 多 个 子 对 话 框 。 在 Android 应 用 中 ， 可 以 使 用 AlertDialog.Builder 对 话 框 ， 开 
发 出 有 多 个 选项 的 对 话 框 。 在 本 实例 中 ， 将 使 用 一 个 按钮 事件 ， 触 发 按钮 事件 后 ， 将 通过 类 
似 列 表 项 目的 方式 呈现 在 Alert Dialog 中 。 上 述 做 法 可 以 实现 常见 的 投票 处 理 、 事 物 选 择 处 理 
各 探 器 等 应 用 。 
3.19.2 具体 实现 

本 节 实 例 保 存在 “ 光 稻 :3\xuanze\” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 

主 文件 src/xuanze/xuanze.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功 
能 。 文 件 xuanze.java 的 具体 实现 代码 如 下 。 


' 


UT 


public class xuanze extends Activity 


{ 
public Button mButton1; 
public TextView mTextView!]; 


/** Called when the activity is first created. */ 
(@Override 
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public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


mButtonl =(Button) findViewById(R.id.myButton!); 
ImTextViewl = (TextView) fndViewById(R.id.myTextViewl); 
mButtonl.setOnClickListener(myShowAlertDialog); 


Button.OnClickListener myShowAlertDialog = new Button.OnClickListener() 
{ 
public void onClick(View arg0) 
{ 
new AlertDialog.Builder(xuanze.this) 
.SetTitle(R.string.str alert title) 
.SetItems(R.array.items irdc dialog, 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog, int whichcountry) 
{ 
CharSequence strDialogBody = getString(R.string.str_ alert body); 
String[] aryShop = 
getResources().getStringArray(R.array.items _irdc dialog); 
new AlertDialog.Builder(xuanze.this) 
.SetMessage(strDialogBody + aryShop[whichcountry]) 
.SetNeutralButton(R.string.str_ ok, new DialogInterface.OnClickListener() 
{ 


public void onClick(DialogInterface dialog, int whichButton) 


{ 


.Show(); 


D) 


.setNegativeButton(" ", new DialogInterface.OnClickListener() 


{ 
(QOverride 
public void onClick(DialogInterface d, int which) 


{ 
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d.dismiss(); 
】 
) 
.Sshow(); 
} 入 结束 onClick 方法 */ 
的 
) 
在 上 述 代 码 中 ， 单 击 按钮 后 会 弹出 选择 。 调 试 运行 后 的 效果 如 图 3-32 所 示 ; 当 单 击 “ 单 

击 后 ”开始 选择 按钮 后 会 弹出 选择 菜单 , 如 图 3-33 所 示 ; 选择 一 个 选项 后 会 弹出 一 个 对 话 框 ， 
如 图 3-34 所 示 。 再 次 单 击 “确认 ”按钮 ， 返 回 图 3-32 所 示 的 初始 界面 。 


团 面 3:45w 


| @ 按 下 开始 选择 | 你 选择 的 是 : C++ 
Ct 确认 


Visual Basic 


图 3-32 运行 结果 图 3-33 ”选项 框 图 3-34 ”选择 提示 
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主题 变换 是 手机 程序 中 的 常见 应 用 之 一 ， 在 一 些 手 机 中 可 以 选择 自己 需要 的 风格 主题 。 
本 节 将 通过 一 个 简单 实例 的 实现 ， 介 绍 Android 实现 主题 变换 的 方法 。 


3.20.1 设计 理念 

在 前 面 的 内 容 中 ， 介 绍 过 Style 的 使 用 方法 。 通 过 使 用 Style， 可 以 快速 开发 出 不 同 外 观 
的 效果 。 通 过 使 用 Style， 还 可 以 大 大 方便 对 程序 的 维护 。Style 的 引入 ， 改 善 了 程序 员 和 视觉 
设计 人 员 之 间 存 在 的 沟通 问题 ,解决 了 J2EE 和 Windows Mobile 中 存在 已 久 的 问题 。 在 
Android 中 ， 除 了 可 以 使 用 Style 定制 外 ， 还 可 以 针对 每 个 Activity、 前 景 、 背 景 和 透明 度 等 进 
行 设置 规划 。 本 节 下 面 介绍 的 实例 ， 是 基于 Style 的 简易 Theme 〈 主 题 ) 应 用 。 


3.20.2 ”具体 实现 
本 节 实 例 保存 在 “光盘 :3\zhutA” 文 件 夹 内 ， 下 面 简单 介绍 主要 文件 的 具体 含义 。 
1. 主 文件 
主 文 件 src/zhuti/zhuti.java 是 此 项 目的 主要 文件 ， 调 用 各 个 公用 文件 来 实现 具体 的 功能 。 
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文件 zhutijava 的 具体 实现 代码 如 下 。 


package irdc.zhuti; 


import irdc.zhuti.R; 

import android.app.Activity; 
import android.os.Bundle; 

public class zhuti extends Activity 


{ 
/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
Wt 
* 应 用 透明 背景 的 主题 
* setTheme(R.style.Theme Transparent); 
WY 
/* 
* 应 用 布景 主题 1 
2 
setTheme(R.style.Theme Translucent); 
* 应 用 布景 主题 2 
* setTheme(R.style.Theme Translucent2); 
/| 
setContentView(R.layout.main); 
} 
} 


在 上 述 代 码 中 , 利用 setTheme 方法 指定 了 Activity 的 主题 ,其 中 主题 设置 文件 在 Stylexml 中 。 
2. 主题 设置 文件 
主题 设置 文件 Stylexml 中 已 经 预先 设置 好 了 Theme 、ThemeTranslucent 、 
ThemeTransparent、TextAppearance.Theme.PlaneText 四 种 主题 样式 。 
调试 运行 后 的 效果 如 图 3-35 所 示 。 


为 深 绿 
背景 应 用 主题 
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手机 项 目 开发 几乎 都 需要 通过 组 件 来 实现 。 组 件 就 如 同一 个 模块 ， 通 过 调用 这 些 组 件 能 
够 实现 对 应 的 功能 效果 。 在 本 章 的 内 容 中 ， 将 通过 具体 的 实例 来 详细 讲解 Android 中 各 个 主 
要 组 件 的 基本 知识 。 希 望 读者 能 够 在 学 习 中 举一反三 ， 为 学 习 本 书后 面 知识 打下 基础 。 


4.1 Editrext]] setoOnkeylstenerUUDOOOODODO 


在 本 节 的 实例 中 ,将 演示 使 用 EditText 文本 编辑 组 件 和 setOnKeyListener 监听 按钮 事件 实 
现 文本 处 理 的 功能 。 本 实例 将 以 EditText 组 件 和 TextView 组 件 来 演示 如 何在 捕捉 用 户 键盘 输 
入 文字 的 同时 ， 实 时 取得 文字 ， 同 步 显 示 于 TextView 中 ， 类 似 手 机 版 的 Ajax 效果 ， 实 时 输 
入 实时 输出 。 

本 实例 实现 代码 保存 在 “光盘 :\daimaM\examplel1”， 下面 开始 讲解 本 实例 的 具体 实现 流程 

在 主 程序 文件 examplel.java 中 ， 关 键 之 处 是 利用 EditText.OnKeyListener 来 拦截 EditText 
的 键盘 输入 事件 ， 只 需 在 其 中 重 写 onKey0 方 法 即 可 实现 。 在 onKey0 方 法 中 ， 将 
EditText.getText() 取出 的 文字 ， 显 示 于 TextView 当中 ， 是 一 个 简单 易 懂 的 范例 练习 。 文 件 
examplel.java 的 具体 实现 代码 如 下 。 


package irdc.examplel; 


0 


PA 


import irdc.examplel.R; 

import android.app.Activity; 
import android.os.Bundle; 
import android.view.KeyEvent; 
import android.view.View; 
import android.widget.EditText; 
import android.widget.TextView; 


public class examplel extends Activity 
{ 
/* 声 明 TextView、EditText 对 和 象 */ 
private TextView mTextView01; 
private EditText mEditTextO1; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 
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super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 取 得 TextView、EditTextx/ 
mTextView01 = (TextView)findViewById(R.id.myTextView); 
mEditText01 = (EditTexbfindViewById(R.id.myEditTextb); 


/* 设 置 EditText 用 OnKeyListener 事件 来 启动 */ 
mEditText01.setOnKeyListener(new EditText.OnKeyListener() 
{ 

@Override 

public boolean onKey(View arg0, int argl, KeyEvent arg2) 


{ 
/# 设 置 TextView 显示 EditText 所 输入 的 内 容 */ 
mTextView01.setText(mEditText01.getText()); 
return false; 


执行 后 的 效果 如 图 4-1 所 示 ， 当 在 文本 杠 输 入 字符 后 ， 在 下 方 会 即时 显示 文本 框 内 输入 


的 字符 ， 如 图 4-2 所 示 。 


这 是 practiceT 


how Me you examplel 


图 4-1 初始 效果 


how are 站 example1 


11111 


证 


图 4-2 即时 显示 提示 信 ， 


上 述 的 实时 输入 实时 显示 效果 可 以 扩展 在 许多 手机 应 用 程序 中 ， 读 者 可 以 在 方法 
OnKeyListener0 里 做 实时 文字 过 滤 效 果 。 例 如 ， 当 用 户 输入 不 雅 的 文字 时 ， 可 以 提示 用 户 不 
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接受 部 分 关键 字 ， 以 输入 “Shit” 为 例 ， 在 TextView 就 会 出 现 : Sh*t， 此 种 做 法 可 以 过 滤 掉 


不 雅文 字 。 


此 外 , 不 仅 可 以 重 写 Widget 中 组 件 的 setOnKeyListener0 方 法 , 而 且 也 可 以 重 写 View ( 视 
图 ) 组 件 中 的 View.setOnKeyListener0 方 法 ， 这 个 方法 能 够 捕捉 用 户 点 击 键盘 时 的 事件 。 在 此 
需要 特别 注意 ， 在 使 用 这 些 方法 时 需要 拦截 这 个 事件 ， 也 就 是 说 当 View 要 取得 Focus (焦点 ) 


时 才能 触发 onoKeyDown 〈 按 键 按 下 时 ) 


Wihal 


oy 


件 。 


全 辣 了 


在 现实 应 用 中 ， 有 时 为 了 特殊 需要 ， 会 需要 设计 


UL 


基于 ImageButton (图 像 按 钮 来 实现 的 。 


个 具有 背景 图 的 按钮 ， 本 节 的 


实现 原理 如 下 。 首 先 ， 将 按钮 背景 图 预先 加 载 至 Drawable 文件 夹 里 (*.png 格式 
文件 )， 利 用 这 些 图 片 ， 作 为 图 片 按 钮 的 背景 图 ， 然 后 ， 


要 设置 图 片 按钮 背景 图 有 许多 方法 ， 此 程序 使 用 的 方法 


实例 是 


的 图 形 


在 布局 中 配置 一 个 “一 般 按钮 ” 读 


者 可 以 查看 两 者 的 对 照 效 果 ， 在 运行 效果 中 ， 可 以 明显 看 出 图 片 按钮 与 一 般 按钮 在 外 


需要 传递 的 参数 即 是 “res/drawable/” 
程序 需要 月 


件 ， 最 后 通过 TextView 来 显示 有 目 


让 用 户 有 动态 交互 的 感觉 。 


在 本 节 的 实例 中 ， 将 演示 使 用 ImageButton 组 件 实现 背景 图 片 按钮 的 功能 。 本 实 


观 上 的 


是 ImageButton.setImageResource()， 
目录 下 面 的 Resource ID， 除 了 设置 背景 图 片 的 方法 外 ， 
月 到 onFocusChange 焦点 变化 监听 与 onClick〈 单 击 ) 等 作为 按钮 单 击 之 后 的 
前 图 片 按钮 的 状态 为 onClick〈 单 击 )、onFocus《〈 事 件 
获得 焦点 时 发 生 ) 或 offFfocus (事件 在 对 象 离开 焦点 时 发 生 )， 并 且 同 步 更 新 按钮 的 背 


处 理事 
在 对 象 


景 图 ， 


例 将 以 


EditText 和 TextView 来 演示 如 何在 捕捉 用 户 键盘 输入 文字 的 同时 ， 实 时 取得 文字 ， 同 步 显示 
于 TextView， 类 似 手机 版 的 Ajax 效果 ， 实 时 输入 实时 输出 。 
本 实例 实现 代码 保存 在 “光盘 :\daimaM\example2”, 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


在 主 程序 文件 example2.java ， 


， 在 图 片 按钮 上 设置 两 个 监听 事件 : onFocusChangeListener 


与 onClickListener 函数 ， 并 实现 Image Button 图 片 的 置换 。 文 件 example2.java 的 具体 实现 代码 


如 下 。 


package irdc.example2; 
import irdc.example2.R; 
import android.app.Activity; 
import android.os.Bundle; 


import android.view.View; 


放 使 用 OnClickListener 与 OnFocusChangeListener 来 区 分 按钮 的 状态 */ 


import android.view.View.OnClickListener; 

import android.view.View.OnFocusChangeListener; 
import android.widget.Button; 

import android.widget.ImageButton; 

import android.widget. TextView; 
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public class example2 extends Activity 


{ 


* 声 明 三 个 对 象 变量 (图 片 按 钮 ,按钮 ,与 TextView)*/ 


private ImageButton mImageButton1; 


private Button mButton1; 


private TextView mTextView!l; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 

super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


/# 通 过 findViewByld 建构 三 个 对 象 */ 

mlmageButtonl1 =(ImageButton) findViewById(R.id.myImageButton!1); 
mButton1=(Button)findViewById(R.id.myButton!1); 

ImTextViewl = (TextView) fndViewById(R.id.myTextViewl); 


/# 通 过 OnFocusChangeListener 来 应 答 ImageButton 的 onFous 事件 */ 
mlmageButtonl.setOnFocusChangeListener(new OnFocusChangeListener() 


{ 


public void onFocusChange(View arg0, boolean isFocused) 


{ 


/* 若 ImageButton 状态 为 onFocus〈 离 开 焦点 ) 改变 ImageButton 的 图 片 
* 并 改变 textView 的 文字 */ 
if (isFocused==true) 
{ 
mTextView1l.setText(" 图 片 按钮 状态 为 :Got Focus"); 
mlmageButtonl.setImageResource(R.drawable.iconfull); 
} 
/# 若 ImageButton 状态 为 offFocus 改变 ImageButton 的 图 片 
* 并 改变 textView 的 文字 */ 
else 
{ 
mTextViewl.setText(" 图 片 按钮 状态 为 :Lost Focus"); 


mlmageButtonl.setImageResource(R.drawable.iconempty); 


} 
)); 


/* 通 过 onClickListener 来 应 答 ImageButton 的 onClick 事件 */ 
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ImImageButton1.setOnClickListenernew OnClickListener() 
{ 
public void onClick(View v) 
{ 
/# 若 图 片 按钮 状态 为 onClick 改变 图 片 按钮 的 图 片 
* 并 改变 textView 的 文字 */ 
mTextViewl.setText(" 图 片 按钮 状态 为 :Got Click"); 
mlmageButton1.setImageResource(R.drawable.iconfull); 
} 
)); 


/# 通 过 onClickListener 来 应 答 Button 的 onClick 事件 */ 
mButtonl.setOnClickListener(new OnClickListener() 
{ 
public void onClick(View v) 
{ 
此 着 Button 状态 为 onClick 改变 ImageButton 的 图 片 
* 并 改变 textView 的 文字 */ 
ImTextViewl.setText(" 图 片 按钮 状态 为 :Lost Focus"); 
ImImageButton1.setImageResource(R.drawable.iconempty); 
} 
)); 
} 
} 


通过 上 述 代码 , 实现 了 图 片 样式 的 按钮 效果 , 执行 后 的 显示 效果 如 图 4-3 所 示 , 单 击 “ 普 
通 按钮 ”后 的 效果 如 图 4-4 所 示 。 

除了 在 运行 时 用 onFocus() 与 onClick() 事件 来 设置 按钮 背景 图 片 外 ，Android 的 MVC 
设计 理念 , 可 以 让 程序 运行 之 初 就 以 XML 定义 的 方式 来 初始 化 图 片 按钮 的 背景 图 , 仅 需 先 将 
图 片 导 入 “res/drawable” 文 件 夹 即 可 。 


how are you example2 


howare you example2 


普通 样式 


图 4-3 ”初始 效果 
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Toast 是 Android 中 的 提示 对 象 ， 它 是 一 个 简短 的 小 信息 ， 将 要 告诉 所 的 信息， 以 一 个 
浮动 在 最 上 层 的 View 显示 。 显 示 Toast 之 后 ， 静 待 儿 秒 后 便 会 自动 消失 ， 最 常见 的 应 用 就 是 
音量 大 小 的 调整 ， 当 单 击 音量 调整 钮 之 后 ， 会 看 见 跳出 的 音量 指示 Toast 对 象 ， 等 待 调整 完 之 
后 便 会 消失 。 
通过 Toast 的 特性 ， 可 以 在 不 影响 用 户 通话 或 聆听 音乐 情况 下 ， 显 示 要 给 用 户 的 信息 。 这 
样 可 以 在 任何 程序 运行 时 ， 通 过 Toast 的 方式 ， 显 示 运 行 变 量 或 手机 环境 的 概况 。 

在 本 节 的 实例 中 ， 将 使 用 一 个 EditText 组 件 来 接受 用 户 输入 的 文字 ， 当 单 击 按钮 时 ， 将 

EditText 里 的 文字 ， 以 方法 ToastmakeTextO 让 文字 显示 于 Toast 对 象 中 ， 这 段 文字 会 在 显示 一 
段 时 间 后 自动 消失 。 
本 实例 实现 代码 保存 在 “光盘 :\daima\\example3” 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 
在 主 程序 文件 example3.java 中 , 构建 2 个 组 件 , EditText 组 件 与 Button 组 件 , 并 在 Button 
组 件 的 onClick0 方法 中 使 用 Toast 对 象 的 makeText0 方 法 来 显示 输入 的 文字 。 文 件 
example3.java 的 具体 实现 代码 如 下 。 


package irdc.example3; 


| 了 


import irdc.example3.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.text.Editable; 

import android.view.View; 

import android.view.View.OnClickListener; 
import android.widget.Button; 

import android.widget.EditText; 

import android.widget.Toast; 


public class example3 extends Activity 

{ 
/** Called when the activity is first created. */ 
放声 明 两 个 对 象 变 量 (按钮 与 编辑 文字 )*/ 
private Button mButton; 
private EditText mEditText; 


(@Override 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 
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/# 通 过 findViewById0 取 得 对 象 对 
mButton=(Button)findViewById(R.id.myButton); 
mEditText=(EditText)findViewById(R.id.myEditText); 


/* 设 置 onClickListener 给 Button 对 象 聆 听 onClick 事件 */ 
mButton.setOnClickListenernew OnClickListener() 
{ 

Override 

public void onClick(View v) 

{ 


* 


放声 明 字符 串 变 量 并 取得 用 户 输 入 的 EditText 字符 串 
Editable Str; 
Str=mEditText.getText(); 


/ 


信使 用 系统 标准 的 makeText0 方 法 来 产生 Toast 信息 */ 
Toast.makeText( 


example3.this, 
"你 的 愿望 "+Str.toString0+" 已 送 达 宝贝 信箱 "， 
Toast.LENGTH LONG).show(); 


/对 青 空 EditText*/ 
mEditText.setText(""); 
} 
)); 
} 
} 


通过 上 述 代 码 ， 实 现 了 温 艺 祝福 提示 的 效果 ， 执 行 后 的 显示 效果 如 图 4-5 所 示 ， 用 户 可 
以 输入 视 福 语句 ， 单 击 “example3” 按 钮 后 的 效果 如 图 4-6 所 示 。 


宙 硬 丘 3:41AM 


p 
你 好 老 同学 


图 4-5 初始 效果 


Toast 提示 信息 在 显示 一 定时 间 后 会 消失 ， 在 Toast 构造 参数 中 的 第 二 个 参数 为 显示 的 时 
间 常 数 ， 可 设置 为 LENGTH LONG 或 LENGTH SHORT， 前 者 提示 时 间 较 长 ， 后 者 较 短 ， 
作为 传递 makeText0 方 法 的 参数 使 用 。 
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图 4-6 温馨 提示 效果 
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CheckBox 是 一 个 复 选 框 组 件 ， 可 以 供用 户 选择 。 在 本 节 实 例 中 , 通过 CheckBox.setOnChecked 
ChangeListener 在 程序 中 设计 了 3 个 CheckBox 选取 项 ， 分 别 表 示 3 种 物品 列表 ， 当 用 户 勾 选 
其 中 一 个 物品 ， 就 在 TextView 里 显示 已 选择 的 物品 列表 。 

本 实例 实现 代码 保存 在 “光盘 :\daima\\example4” 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 
在 主 程序 文件 example4.java 中 ， 分 别 创建 了 3 个 CheckBox 对 象 和 一 个 TextView 对 象 ， 
通过 监听 setOnCheckedChangeListener〈 当 选项 组 中 的 按钮 的 勾 选 状态 发 生 改变 时 的 监听 事 
件 ) 事件 ， 利 用 onCheckedChanged() 方 法 来 更 新 TextView 文字 。 文 件 example4.java 的 具体 实 
现代 码 如 下 。 


package irdc.example4; 


import irdc.example4.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.widget.CheckBox; 
import android.widget.CompoundButton; 
import android.widget.TextView; 


public class example4 extends Activity 
放声 明 对 象 变 量 */ 
private TextView mTextView!; 
private CheckBox mCheckBox!; 
private CheckBox mCheckBox2; 
private CheckBox mCheckBox3; 
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/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


/# 通 过 findViewById 取得 TextView 对 象 并 调整 文字 内 容 */ 
ImTextViewl = (TextView) fndViewById(R.id.myTextViewl); 
mTextView1.setText(" 你 所 选择 的 项 目 有 : "); 


/# 通 过 findViewByld 取得 三 个 CheckBox 对 象 */ 

mCheckBox1=(CheckBox)findViewById(R.id.myCheckBox!1); 
mCheckBox2=(CheckBox)findViewById(R.id.myCheckBox2); 
mCheckBox3=(CheckBox)findViewById(R.id.myCheckBox3); 


/# 设 置 OnCheckedChangeListener 给 三 个 CheckBox 对 象 */ 

ImCheckBoxl.setOnCheckedChangeListener(mCheckBoxChanged); 
mCheckBox2.setOnCheckedChangeListener(mCheckBoxChanged); 
mCheckBox3.setOnCheckedChangeListener(mCheckBoxChanged); 


放声 明 并 建构 onCheckedChangeListener 对 铺 */ 
private CheckBox.OnCheckedChangeListener mCheckBoxChanged 
= new CheckBox.OnCheckedChangeListener() 
{ 
/*implement onCheckedChanged 方法 */ 
(QOverride 
public void onCheckedChanged(CompoundButton buttonView, 
boolean isChecked) 


/# 通 过 getString0 取 得 CheckBox 的 文字 字符 串 */ 
String str0=" 所 选 的 项 目 为 : "; 

String strl=getString(R.string.str_checkbox1); 
String str2=getString(R.string.str_ checkbox2); 
String str3=getString(R.string.str_ checkbox3); 
String plus=";"; 

String result=" 但 是 超过 预算 喝 !1"; 

String result2=" 还 可 以 再 多 买 几 本 喔 !1"'; 


放任 一 CheckBox 被 勾 选 后 ,该 CheckBox 的 文字 会 改变 TextView 的 文字 内 容 
* 三 个 对 象 总 共 八 种 情况 */ 


加 国 89 


9O 国 国 


Android 开发 完全 实战 宝典 


if{mCheckBox1.isChecked()==true & mCheckBox2.isChecked()==true 
& mCheckBox3.isChecked()==true) 


mTextViewl.setText(str0+strl+pluststr2+plust+str3+result); 
} 
else if{mCheckBox!.isChecked()==false & mCheckBox2.isChecked()==true 
& mCheckBox3.isChecked()==true) 


mTextViewl.setText(str0+str2+pluststr3+result); 
} 
else if{mCheckBox!.isChecked()==true & mCheckBox2.isChecked()==false 
& mCheckBox3.isChecked()==true) 


mTextViewl.setText(str0+strl+pluststr3+result); 
} 
else if{mCheckBox1.isChecked()==true & mCheckBox2.isChecked()==true 
& mCheckBox3.isChecked()==false) 


mTextViewl.setText(str0+strl+pluststr2+result); 
} 
else 1f{mCheckBox].isChecked()==false & mCheckBox2.isChecked()==false 
& mCheckBox3.isChecked()==true) 


mTextViewl.setText(str0+str 3+plustresult2); 
} 
else 1f{mCheckBox].isChecked()==false & mCheckBox2.isChecked()==true 
& mCheckBox3.isChecked()==false) 


mTextView!l.setText(strO+str2); 
} 
else 1f{mCheckBox1.isChecked()==true & mCheckBox2.isChecked()==false 
& mCheckBox3.isChecked()==false) 


mTextViewl.setText(strO+str1); 


} 
else 1f{mCheckBox1.isChecked()==false & mCheckBox2.isChecked()==false 


& mCheckBox3.isChecked()==false) 


mTextViewl.setText(str0); 
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在 上 述 代 码 中 ， 首 先 设 置 一 共 拥有 3600 块 钱 ， 然 后 提供 了 一 个 货物 清单 供用 户 选择 要 

买 的 商品 。 当 用 户 选择 商品 后 ， 会 显示 对 应 的 购买 物品 ， 如 果 超 出 了 3600 这 个 额度 ， 会 显 

:出 对 应 的 提示 。 

实例 执行 后 的 初始 效果 如 图 4-7 所 示 ， 当 选择 一 种 商品 后 ， 在 下 方 会 显示 出 对 应 的 提示 ， 
如 图 4-8 所 示 。 如 果 超 出 了 预算 3600， 也 会 显示 对 应 的 提示 ， 如 图 4-9 所 示 。 


闭 丘 3:46 AN 团 挟 3:47 AM 


| 油 


~ 


图 iPhone 手 机 图 iPhone 手 机 


国 caH-* 


尔 所 选择 的 项 目 有 : 


图 -和 村 一 本 
所 选 的 项 目 为 : C++ 教材 一 本 ;还 可 以 再 多 买 几 本 喔 ! 


图 4-7 初始 效果 图 4-8 显示 提示 效果 


iPhone 手 机 
C++ 教材 一 本 


所 选 的 项 目 为 : 电脑 ;iPhone 手机 ;C++ 教材 一 本 但 是 
超过 预算 叫 由 


图 4-9 超出 预算 提示 效果 


pnt 


为 了 满足 特殊 需求 ， 可 以 将 OnCheckedChangeListener 改 为 OnTouchListener 〈 屏 幕 触 控 
事件 )， 有 具体 实现 代码 如 下 。 


private CheckBox.OnTouchListener mCheckBoxTouch = 
new CheckBox.OnTouchListener() 
{ 
(@Override 
public boolean onTouch(View v, MotionEvent event) 
{ 
谨 判断 在 触 控 笔 指 压 此 组 件 时 的 状态 * 
if(mCheckBox! .isChecked()==false) 
{ 
旋 当 触 控 笔 放 开 后 的 动作 */ 
} 
else if(mCheckBox!1.isChecked()==true) 
{ 
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/# 当 触 控 笔 压 下 后 的 动作 关 
} 
return false; 


} 
扎 


当 用 户 使 用 某 些 网 络 服务 时 ， 系 统 会 要 求 用 户 同意 某 些 条 款 。 在 手机 应 用 程序 、 手 机 游 
戏 的 设计 过 程 中 ， 也 会 常 遇见 CheckBox 在 同意 条 天 情境 的 使 用 ， 其 选取 的 状态 有 两 种 : 


isChecked=true 与 isChecked=false。 


在 本 节 实 例 中 ， 使 用 TextView 放 入 条 球 文字 ， 在 下 方 配置 一 个 CheckBox Widget 作为 选 
取 项 ， 通 过 Button.onClickListener 按钮 事件 处 理 ， 取 得 用 户 同意 条 球 的 状态 。 
当 CheckBox.isChecked 为 tue， 更 改 TextView 的 文字 内 容 为 “你 已 接受 同意 !!” 当 未 选 
取 CheckBox 时 ，Button 是 不 可 以 选择 的 (被 Disabled)。 


本 实例 实现 代码 保存 在 “光盘 :\daima\4\example5”， 下面 开 始 讲解 本 实例 的 具体 实现 


里 


] CheckBox.OnClickListener 〈 单 击 监听 ) ! 


在 主 程序 文件 example5.java 中 ， 利 


体 实现 代码 如 下 。 


package irdc.example5; 


import irdc.example5.R; 

import android.app.Activity; 
import android.os.Bundle; 

import android.view.View; 
import android.widget.Button; 
import android.widget.CheckBox; 
import android.widget. TextView; 


public class exampleS extends Activity 


{ 


/** Called when the activity is first created. */ 


/* 声 明 TextView、CheckBox、Button 对 象 */ 


public TextView myTextView!l; 
public TextView myTextView2; 
public CheckBox myCheckBox; 
public Button myButton; 


(QOverride 


public void onCreate(Bundle savedInstanceState) 
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的 事件 来 


判断 Button 该 不 该 显示 , 其 方法 就 是 判断 Button.Enabled (按钮 是 否 选择 ) 的 值 ; 在 一 开始 时 ， 
默认 参数 为 false， 当 单 击 CheckBox 时 ，Button 参数 就 修改 为 tue。 文 件 example5.java 的 具 
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super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


/* 取 得 TextView、CheckBox、Button*/ 

myTextViewl = (TextView) findViewById(R.id.myTIextViewl); 
myTextView2 = (TextView) findViewById(R.id.myTextView2); 
myCheckBox = (CheckBox) fndViewById(R.id.myCheckBox); 
myButton = (Button) fndViewById(R.id.myButton); 


/* 将 CheckBox、Button 默认 为 未 选择 状态 */ 
myCheckBox.setChecked(false); 
myButton.setEnabled(false); 


myCheckBox.setOnClickListener(new CheckBox.OnClickListener() 
{ 
@Override 
public void onClick(View v) 
{ 
if{myCheckBox.isChecked()) 
{ 
伴 设置 按钮 为 不 能 选择 对 象 凡 
myButton. setEnabled(true); 
myTextView2.setText(""); 
} 
else 
{ 
伴 设置 按钮 为 可 以 选择 对 象 沁 
myButton.setEnabled(false); 


ImyTextView1.SetText(R.string.textl ); 
/* 在 TextView2 里 显示 出 "请 勾 选 我 同意 "*/ 
myTextView2.setText(R.string.no); 


} 
外 


myButton.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
if(myCheckBox.isChecked()) 
{ 


myTextViewl].setText(R.string.ok); 
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}else 
} 
yr 
); 


} 
} 


实例 执行 后 的 初始 效果 如 图 4-10 所 示 ， 当 选择 “同意 ”选项 并 单 击 “确定 ”按钮 后 ， 会 


输出 对 应 的 提示 界面 ， 如 图 4-11 所 示 。 


tn 
注册 之 前 认真 阅读 全 部 层 
如 有 任何 


容 ， 
问 ， 避风 本 册 站 和 无 论 
您 事实 上 是 之 前 认 
阅读 本 交 血 角 讽 ， 


技 照 本 网 站 注册 程序 成 功 注册 
为 用 户 、 您 的 行为 仍然 表示 你 


我 同意 


图 4-10 ”初始 效果 图 


当 CheckBox 在 默认 内 容 为 空白 时 〈 没 有 任何 默认 的 提示 文字 
字 ， 其 调用 的 方法 为 CheckBox.setHint0 方 法 。 


2 加 本 用 加 加 国医 同 区 吕 本 册 加 站 


单 选 按钮 组 能 够 将 不 同 的 单 选 按钮 布置 于 同一 个 单 选 按钮 组 ， 


4-11 输出 提示 
下 )， 可 设置 提示 用 户 的 文 


同属 一 个 单 选 按钮 组 里 的 


按钮 ， 只 能 做 出 单一 选择 ， 昌 然 前 一 章 曾 经 介绍 过 单 选 按钮 组 与 单 


选 按钮 ， 但 当时 使 用 的 是 


按钮 事件 ， 在 此 要 示范 “ 单 击 ”的 同时 就 运行 事件 处 理 ， 不 再 需要 按钮 的 辅助 了 。 

在 本 节 实 例 中 ， 先 设计 一 个 TextView Widget 和 一 个 单 选 按钮 组 ， 并 在 单 选 按钮 组 内 放置 
2 个 单 选 按钮 ， 默 认为 不 选择 。 在 程序 运行 阶段 ， 利 用 onCheckedChanged (选择 改变 事件 ) 
作为 启动 事件 装置 ,让 用 户 选择 其 中 一 个 按钮 时 ， 显 示 被 选择 的 内 容 , 最 后 将 RadioGroup ( 单 


选 按钮 ) 的 选项 文字 显示 于 TextView 当中 。 


本 实例 实现 代码 保存 在 “光盘 :daimav4\example6” 下 面 开 始 讲 
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解 本 实例 的 具体 实现 流程 。 


的 


件 , 随后 将 被 选择 的 单 选 


> 


在 主 程序 文件 example6.java 中 ， 使 用 OnCheckedChangeListener 来 启动 RadioGroup 的 奸 
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具体 实现 代码 如 下 。 


oy 


按钮 (mRadiol.getText( ) 的 文字 显示 于 TextView。 文 们 


package irdc.example6; 


import irdc.example6.R; 


import android.app.Activity; 


import android.os.Bundle; 


import android.widget.RadioButton; 


import android.widget.RadioGroup; 


import android.widget.TextView; 


public class example6 extends Activity 


{ 


public TextView mTextView!]; 


public RadioGroup mRadioGroupl; 
public RadioButton mRadiol,mRadio2; 


/** Called when the activity is first created. */ 


(@Override 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


人 # 取 得 TextView、RadioGroup、RadioButton 对 象 的 内 容 */ 


mTextView!l 


= (TextView) fmndViewById(R.id.myTextView); 


mRadioGroupl = (RadioGroup) findViewById(R.id.myRadioGroup); 
mRadiol = (RadioButton) fndViewById(R.id.myRadioButtonl ); 
mRadio2 = (RadioButton) findViewById(R.id.myRadioButton2); 


人/# 单 选 按钮 组 用 OnCheckedChangeListener 来 运行 */ 
ImRadioGroup1.setOnCheckedChangeListener(mChangeRadio); 


private RadioGroup.OnCheckedChangeListener mChangeRadio = new 
RadioGroup.OnCheckedChangeListener() 


(QOverride 


public void onCheckedChanged(RadioGroup group, int checkedId) 


{ 


if(checkedId==mRadiol1.getId()) 


{ 


/* 把 mRadiol 对 象 的 内 容 传 到 mTextView1*/ 


F example6.java 
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mTextViewl.setText(mRadiol.getText()); 


} 


else if(checkedId==mRadio2.getId()) 


{ 


/* 把 mRadio2 对 象 的 内 容 传 到 mTextView1*/ 
mTextViewl.setText(mRadio2.getText()); 


实例 执行 后 的 初始 效果 如 图 4-12 所 示 ， 当 选择 一 个 选项 后 会 显示 出 选择 的 值 , 如 图 4-13 


所 示 。 


磺 关 全 4:00A 


图 4-12 初始 效果 


4.6 ImageView[| 口 DOO 


在 本 节 实 例 中 ， 事 先 准 备 了 三 张 图 片 ( 两 张 外 框图 、 


“res/drawable” 文件 夹 下 面 在 此 使 用 的 图 片 为 PNG 图 形 文件 ， 而 图 案 大 小 调整 成 了 手机 屏 


幕 大 小 ， 当 然 也 可 以 依据 手机 的 分 辨 率 ， 


图 4-13 输出 提示 


一 张 内 框图 )， 将 这 三 张 图 片 放 在 


动态 调整 ImageView〔 图 片 组 件 ) 的 大 小 。 


Fe 


司 


然后 在 布局 当中 创建 了 两 个 ImageView， 且 以 绝对 定位 的 方式 “堆积 ”在 一 起 ， 在 其 下 


方 放 上 两 个 按钮 ， 按 钮 的 目的 是 为 了 要 用 来 切换 


置换 图 片 的 动作 。 


当 单 击 Buttonl 后 ，ImageView1l 会 出 现 right 的 图 片 ; 
会 出 现 left 的 图 片 ， 而 ImageView2 对 和 象 丝 为 固定 不 动 ( 文 件 


图 片 ， 创 建 完 成 后 ， 要 在 Button 事件 里 处 理 


名 叫 0a)。 


击 Button2 后 ，ImageViewl 对 象 


早 避 


本 实例 实现 代码 保存 在 “光盘 :\daima\\example7” 下 面 开 始 讲解 本 实例 的 有 具体 实现 流程 。 
在 主 程序 文件 example7.java 中 ， 其 核心 功能 是 通过 getResources0 方 法 实现 的 ， 此 方法 负 


责 访 问 Resource ID ， 访 问 资 源 里 的 图 文 从 
getResources().getDrawable() 来 载 入 “res/drawable ”日 53 


ImageView 当中 。 文 件 example7.java 的 


package irdc.example7; 


import irdc.example7.R; 
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EF、 文字 都 要 用 
录 里 的 图 文件 ， 并 将 图 片 放置 
具体 实现 代码 如 下 。 


到 getResources0; 在 此 使 用 


RN 
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import android.app.Activity; 


import android.os.Bundle; 


import android.view.View; 


import android.widget.Button; 


import android.widget.ImageView; 


public class example7 extends Activity 


{ 


放声 明 Button、ImageView 对 象 */ 


private ImageView mImageView0!1; 


private ImageView mImageView02; 


private Button mButton01; 


private Button mButton02; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


人 # 取 得 Button、ImageView 对 象 */ 

mlmageView01 = (ImageView)findViewById(R.id.myImageView1l); 
mlmageView02 = (ImageView)findViewById(R.id.myImageView2); 
mButton01 = (Button) findViewById(R.id.myButton!1); 

mButton02 = (Button) findViewById(R.id.myButton2); 


人 # 设 置 ImageView 背景 图 */ 


mlmageView01.setImageDrawable(getResources(). 
getDrawable(R.drawable.right)); 

mlmageView02.setImageDrawable(getResources(). 
getDrawable(R.drawable.o0a)); 


/* 用 OnClickListener 事件 来 启动 */ 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 

Override 

public void onClick(View V) 

{ 
访 当 启动 后 ，ImageView 立刻 换 背 景 图 */ 
mlmageView01.setImageDrawable(getResources(). 
getDrawable(R.drawable.right)); 


未 
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mButton02.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 

{ 
mlmageView01.setImageDrawable(getResources(). 
getDrawable(R.drawable.left)); 


); 


实例 执行 后 的 初始 效果 如 图 4-14 所 示 ， 当 分 别 单 击 按钮 “pic1” 和 “pic2” 后 ， 会 分 别 
显示 不 同 素材 的 相框 ， 如 图 4-15 所 示 。 


RH 


图 4-14 ”初始 效果 图 4-15 不 同 素材 相框 


对 于 上 述 实例 ， 读 者 可 以 将 两 个 InageButton〈 图 片 按钮 ) 堆放 在 一 起 ， 如 此 一 来 ， 不 但 
有 背景 图 ， 而 且 还 有 按钮 事件 可 以 触发 。 代 码 如 下 。 


<ImageButton 
android:id="(@+id/myImageButton1" 
android:state focused="true" 
android:layout width="320px" 
android:layout_height="280px" 
android:layout x="O0px" 
android:layout_ y="36px" 

/这 


ImageButton 的 使 用 方法 比较 简单 ， 而 堆栈 的 技巧 可 参考 这 个 范例 程序 ， 比 较 不 同 的 地 方 
就 是 只 要 单 击 图 片 ， 即 可 直接 做 换 图 的 动作 ， 不 需要 再 单 击 下 面 的 Button 做 更 换 ， 需 要 注意 
98 看 下 
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的 是 图 片 大 小 要 作 调 整 ， 否 则 可 能 会 与 InageButton 不 合 。 


4 


4.7 Spinne0OODODODO 


Spinner 是 一 个 拉 菜 和 


面 大 小 有 限 ， 所 以 要 在 有 限 的 范 


意 ， 类 


在 本 节 实 例 


以 一 段 动画 提示 用 户 。 


BR 


， 将 自 定 义 下 拉 沫 单 里 上 
下 拉 菜 单 的 显示 方式 ) 方法 ， 以 XML 的 方式 定义 下 拉 沫 单 
下 拉 荣 单 ， 还 用 程序 设计 了 一 段 动画 ， 当 月 


似 于 Swing 
围 选择 项 目 ， 下 拉 来 六 
的 样式 ， 然 后 调 


全 
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的 ComboBox 、HTML 的 <select>。 因 为 手机 面 


是 1 


] setDropDownViewResource〈 设 置 


、 也 是 较 好 的 选择 。 


要 显示 的 模样 。 实 例 中 除了 自 定义 
昌 户 以 触 控 的 方式 点 击 这 个 自 定义 的 Spinner 时 ,会 


本 实例 实现 代码 保存 在 “光盘 :\daima\\example8” 下 面 开 始 讲解 本 实例 的 有 具体 实现 流程 。 
在 主 程序 文件 example8.java 中 ,在 新 建 ArrayAdapter( 数 据 视 图 对 象 时 ) 使 用 ArrayAdapter 


(Context context, int textViewResourceId, T[] objects) 这 个 Constructor，textViewResourceld 使 用 


Android 提供 的 ResourceID ，objects 为 必须 传递 的 字符 


example8.java 的 具体 实现 代码 如 下 。 


package irdc.example8; 


import irdc.example8.R; 


import android.app.Activity; 


import android.os.Bundle; 


import android.view.MotionEvent; 


import android.view.View; 


import android.view.animation.Animation; 


import android.view.animation.AnimationUtils; 


import android.widget.AdapterView; 


import android.widget. ArrayA dapter; 


import android.widget. TextView; 


import android.widget.ListView; 


import android.widget.Spinner; 


public class example8 extends Activity 


{ 


private static final String[] countriesStr = 


人 


Zu MlE 


目 "， "法 转 


private TextView myTextView; 


private Spinner mySpinner; 


六 


private ArrayA dapter<String> adapter; 


Animation myAnimation; 


/** Called when the activity is first created. */ 


(@Override 


public void onCreate(Bundle savedInstanceState) 


{ 


数组 (String Array)。 文 件 
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super.onCreate(savedInstanceState); 
/+ 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


/# 以 findViewById0 取 得 对 象 */ 
myTextView = (TextView) findViewById(R.id.myTextView); 
mySpinner = (Spinner) findViewByld(R.id.mySpinner); 


adapter =new ArrayAdapter<String>(this, 
android.R.layout.simple_ spinner item, countriesStr); 

/*myspinner_dropdown 为 自 定义 下 拉 菜 单 模式 定义 在 res/layout 目录 下 对 

adapter.setDropDownViewResource(R.layout.myspinner dropdown); 


/* 将 ArrayAdapter 添加 Spinner 对 象 中 */ 
mySpinner.setAdapter(adapter); 


/+ 将 mySpinner 添加 OnItemSelectedListener */ 
mySpinner.setOnItemSelectedListener 
(new Spinner.OnltemSelectedListener() 

{ 

Override 

public void onItemSelected 

(AdapterView<?> arg0, View argl, int arg2, 
long arg3) 


/* 将 所 选 mySpinner 的 值 带 入 myTextView 中 */ 
ImyTextView.setText(" 选 择 的 是 " + countriesStr[arg2]); 
/# 显示 mySpinner 对 象 的 内 容 */ 
arg0.setVisibility(View.VISIBLE); 


(QOverride 
public void onNothingSelected(AdapterView<?> arg0) 


{ 


2 


/# 取得 Animation 定义 在 res/anim 目录 下 */ 


myAnimation = AnimationUtils.load Animation(this, R.anim.my_anim); 


/* 将 为 mySpinner 添加 OnTouchListener〈 触 摸 监听 事件 ) */ 
mySpinner.setOnTouchListener(new Spinner.OnTouchListener() 


{ 


(QOverride 
public boolean onTouch(View v, MotionEvent event) 


10O 国 轩 
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/* 将 mySpinner 运行 Animation 效果 的 动画 */ 
Vv.startAnimation(myAnimation); 

/# 将 mySpinner 对 象 隐藏 */ 
Vv.setVisibility(View.INVISIBLE); 

return false; 


| 


mySpinner.setOnFocusChangeListener(new Spinner.OnFocusChangeListener() 
{ 
(QOverride 
public void onFocusChange(View v, boolean hasFocus) 
{ 
// TODO Auto-generated method stub 
} 
); 
} 
) 


Adapter 的 setDropDownViewResource 可 以 设置 下 拉 荣 单 的 显示 方式 ， 将 该 XML 定义 在 
res/layout 目录 下 面 ， 可 针对 下 拉 菜 单 中 的 文本 框 进 行 设 置 ， 如 同 本 程序 里 的 
R.layout.myspinner_dropdown 即 为 自 定义 的 下 拉 荣 单 文 本 框 样式 。 除 了 改变 下 拉 沫 单 样式 外 ， 
也 对 Spinner 做 了 一 点 动态 效果 ， 单 击 下 拉 菜 单 时 ， 晃 动 下 拉 沫 单 再 出 现下 拉 菜 单 
(myAnimation ) 。 

执行 后 的 初始 效果 如 图 4-16 所 示 ， 单 击 下 拉 荣 单 后 会 显示 一 个 悬浮 选项 效果 ， 里 面 显 示 
了 4 个 国家 供用 户 选 择 ， 如 图 4-17 所 示 。 


意大利 
选择 的 是 意大利 英格兰 
意大利 =| 德国 
法 国 
图 4-16 初始 效果 图 4-17 4 个 选项 


当选 择 一 个 选项 后 ， 会 显示 出 对 应 的 提示 信息 ， 例 如 选择 了 “意大利 ” 效果 如 图 4-18 
所 示 。 


example8 


谈 


图 4-18 提示 信 ， 
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48 Gale00000 


在 上 一 章 的 内 容 中 ， 通 过 Gallery〔 图 像 库 组 件 ) 实现 了 一 个 相片 画 亡 的 效果 。 在 本 节 实 
例 中 , 首先 将 6 张 PNG 图 片 导入 Drawable 文件 夹 当 中 , 并 于 onCreate 的 同时 , 载 入 到 Gallery 
Widget 中 ; 然后 ， 添 加 一 个 OnItemClick( 当 单 击 Gallery 中 不 同 的 图 像 时 触发 ) 的 事件 ， 以 
取得 图 片 的 了 D 编号 来 响应 用 户 点 击 图 片 时 的 状态 ， 完 成 Gallery 的 高 级 使 用 。 
本 实例 的 另 一 个 重点 是 设置 Gallery 图 片 的 宽 高 ， 以 及 设置 图 片 Layout 的 大 小 ， 在 此 编 
写 一 个 继承 自 BaseAdapter (基本 容器 ) 的 InageAdapter〈 图 像 构造 器 ) 来 存放 图 片 ， 通 过 
ImageView.setScaleType0 方 法 来 改变 图 片 的 显示 方式 ， 再 通过 setLayoutParams() 方 法 来 改变 
Layout 的 宽 高 。 
本 实例 实现 代码 保存 在 “光盘 Vs 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 
在 主 程序 文件 example9.java 中 有 2 个 重点 需要 注意 。 
1) ImageAdapter 继承 BaseAdapter class 的 未 实现 方法 的 重 写 构造 。 
2) Gallery 的 OnItemClick( 方法 与 图 片 及 Layout 宽 高 设置 。 
文件 example9.java 的 有 具体 实现 代码 如 下 。 


package irdc.example9; 


import irdc.example9.R; 
import android.app.Activity; 
import android.os.Bundle; 


旋 本 范例 需 使 用 到 的 类 六 

import android.content.Context; 

import android.content.res.TypedArray; 
import android.view.View; 


import android.view.ViewGroup; 

import android.widget.AdapterView; 

import android.widget.BaseAdapter; 

import android.widget.Gallery; 

import android.widget.ImageView; 

import android.widget.Toast; 

import android.widget.AdapterView.OnItemClickListener; 


public class example9 extends Activity 
{ 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
/# 通 过 findViewById 取得 */ 
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Gallery g = (Gallery) i id.mygallery); 


/# 添加 一 ImageAdapter 并 设置 给 Gallery 对 象 */ 
g.setAdapter(new eR 


型 - 


放 设 


-个 itemclickListener 并 Toast 〈 提 示 ) 被 单 避 


图 片 的 位 置 * 


二 


g.setOnItemClickListener(new OnltemClickListener() 


{ 
public void onItemClick 


(AdapterView<?> parent, View v, int position, long 1d) 


{ 


Toast.makeText 
(example9.this, getStrin 


+ position+ getString(R. 


g(R.string.my_gallery_ text pre) 
string.my_ gallery_text_post), 


Toast.LENGTH SHORT).show(); 


} 
D); 
} 
/* 改写 BaseAdapter 自 定 义 - 
public class ImageAdapter exte 
{ 
/* 声 明 变 量 */ 


~ ImageAdapter class */ 
nds BaseAdapter 


int mGalleryltemBackground; 


private Context mContext; 


/ImageAdapter 的 构造 器 */ 


public ImageAdapter(Context c) 


{ 


mContext = ¢; 


/# 使 用 在 res/values/attrs 
* 的 Gallery 属性 .*/ 


.xml 中 的 <declare-styleable> 定 义 


TypedArray a = obtainStyledAttributes(R.styleable.Gallery); 


/取得 Gallery 属性 的 Index id (图 像 编号 ) */ 


mGalleryItemBackground = 


a.getResourceld 


(R.styleable.Gallery android galleryltemBackground, 0); 


ol 
讼 


/让 对 象 的 styleable 


arecycle(); 


/#* 履 盖 的 方法 getCount, 返 
public int getCount() 


{ 


生 能 够 反复 使 用 */ 


区 回 图 片 数 目 光 


retum myImageIds.length; 
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这 样 ， 整 个 实例 介绍 完毕 。 执 行 后 的 效果 如 图 4-19 所 示 ， 通 过 滑动 鼠标 可 以 浏览 每 


图 片 。 
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} 


} 


/* 覆盖 的 方法 getItemId, 返 回 图 像 的 数组 id */ 


public Object getItem(int position) 


{ 


return position; 


} 


public long getItemId(int positiom) 


{ 


return position; 


/* 履 盖 的 方法 getView, 返 回 一 View 对 象 */ 
public View getView 


(int position, View convertView, ViewGroup parent) 

{ 
# 产 生 ImageView 对 象 */ 
ImageView i= new ImageView(mContext); 
放 设 置 图片 给 imageView 对 象 */ 
i.SetImageResource(myImageIds[position]); 
旋 重 新 设置 图 片 的 宽 高 */ 
i.setScaleType(ImageView.ScaleType.FIT XY); 
/* 重 新 设置 Layout 的 宽 高 */ 
i.setLayoutParams(new Gallery.LayoutParams(136, 88)); 
/#* 设 置 Gallery 背景 图 */ 
i.setBackgroundResource(mGalleryIltemBackground); 
/返回 imageView 对 象 */ 


return 1; 


放 建 构 一 Integer array 并 取得 预 加 载 Drawable 的 图 片 id*/ 
private Integer[| myImagelds = 
{ 

R.drawable.photo]1, 

R.drawable.photo2, 

R.drawable.photo3, 

R.drawable.photo4, 

R.drawable.photo5, 

R.drawable.photo6， 


FE 
再 可 
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图 4-19 执行 效果 


49 java.io.Hleq DODODOOD 


件 搜索 的 功能 。 在 Java LO 的 API 中 ， 提 供 了 java.io.File 对 象 ， 只 要 利用 File 对 象 的 方法 ， 
了 搭配 Android 的 EditText、TextView 等 组 件 ， 就 可 以 轻松 实现 手机 的 文件 搜索 引擎 。 


通过 文件 搜索 功能 ， 可 以 快速 的 帮 我 们 找到 需要 的 文件 。 同 样 ， 在 手机 中 也 可 以 实现 文 


在 本 节 实 例 中 ， 使 用 EditText、Button 与 TextView 三 种 组 件 来 实现 此 功能 ， 用 户 将 要 搜 


索 的 文件 名 称 或 关键 字 输 入 EditText 中 , 单 击 Button 后 , 程序 会 在 根 目录 中 寻找 符合 的 文件 ， 
并 将 搜索 结果 显示 于 TextView 中 ;如 果 找 不 到 符合 的 文件 ， 则 显示 找 不 到 文件 。 


i 


后 


行 


符 


本 实例 实现 代码 保存 在 “光盘 :\daimaM\example10”， 下面 开始 讲解 本 实例 的 具体 实现 


在 主 程序 文件 example10.java 中 ， 以 java.io.File 对 象 来 取得 根 目 录 下 的 文件 ， 经 过 比较 
将 符合 条 件 的 文件 路 径 写 入 TextView 中 ， 若 要 在 TextView 中 换行 ， 需 使 用 “\n” 作 为 换 
号 。 文 件 example10.java 的 具体 实现 代码 如 下 : 


package irdc.example10; 


/# 加 载 相关 类 */ 


import irdc.example10.R; 


import java.io.File; 

import android.app.Activity; 
import android.os.Bundle; 
import android.view.View; 
import android.widget.Button; 
import android.widget.EditText; 
import android.widget.TextView; 


public class example10 extends Activity 
放声 明 对 象 变 量 */ 
private Button mButton; 
private EditText mKeyword; 
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private TextView mResult; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/# 载 入 布局 文件 main.xml */ 


setContentView(R.layout.main); 


雍 初始 化 对 象 忆 
mKeyword=(EditText)findViewById(R.id.mKeyword); 
mButton=(Button)findViewById(R.id.mButton); 
mResult=(TextView) findViewById(R.id.mResult); 


/* 将 mButton 对 象 添加 onClickListener 事件 */ 
mButton.setOnClickListener(new Button.OnClickListener() 
{ 
public void onClick(View v) 
{ 
放 取 得 输入 的 关键 字 */ 
String keyword = mKeyword.getText().toString(); 
if(keyword.equals("")) 
{ 
mResult.setText(" 关 键 字 不 能 为 空 !1"); 
} 
else 
{ 
mResult.setText(searchFile(keyword)); 
} 


A 


上 # 搜索 文件 的 方法 * 
private String SearchFile(String keyword) 
{ 
String result=""; 
File[] files=new File("/").listFiles(); 
for( File f: files ) 
{ 
if(f.getName().indexOf(keyword)>=0) 
{ 
result+=f.getPath()+"\n"; 


} 
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return result; 
} 
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iftresult.equals("")) result=" 找 不 到 文件 !1"'; 


在 上 述 代 码 中 ，searchFile(String keyword) 方 法 的 功用 是 为 搜索 根 目 录 下 符合 关键 字 的 文 
件 ， 在 搜索 文件 的 过 程 中 ， 只 搜索 根 目录 中 的 文件 ， 并 没有 再 对 子 目 录 下 的 文件 做 进一步 的 


比较 ， 如 果 要 再 强化 这 个 文件 搜索 的 功能 ， 
程序 中 利用 File.isDirectory0 这 个 方法 来 判断 其 是 否 为 目录 。 如 果 是 的 话 
找 ; 不 是 的 话 ， 就 终止 向 下 寻找 的 动作 。 这 个 做 法 在 后 面 的 范例 9 
的 是 手机 硬件 环境 是 否 能 负荷 程序 做 大 规模 的 文人 


存 ) 是 比 不 上 一 般 计算 机 的 。 


让 它 也 能 搜索 包含 子 目 录 下 的 所 有 文件 ， 可 以 在 


， 就 继续 往 下 一 层 寻 


Ph 会 有 详细 的 示范 ， 要 注意 
搜索 ， 毕 竞 手 机 的 硬件 配备 《处理 器 、 内 


执行 后 的 效果 如 图 4-20 所 示 ， 输 入 搜索 关键 字 ， 并 单 击 “搜索 处 理 ” 按 钮 后 会 显示 出 对 
应 的 搜索 结果 ， 如 图 4-21 所 示 。 


得 5554:aa 


实例 10 


图 4-21 搜索 效果 


4.10 ImageButonl DDOODOODO 


Android 的 上 默认 按钮 通常 都 是 方 方 正 正 的 , 在 本 实例 


当 单 击 A 按钮 ， 恢 复 B 按钮 


图 片 ， 当 单 击 B 按钮 ， 恢 复 A 按钮 的 图 片 。 使 月 


FP 可 以 实现 两 个 按钮 之 间 的 交互 ， 即 


的 方法 为 单 击 


的 瞬间 置换 图 片 ， 置 换 的 图 片 方 式 与 先前 介绍 的 相同 (ImageButton.setImageDrawable)。 


1. 实现 原理 


在 实例 中 使 用 了 3 张 图 片 ,分 别 是 pl.png、p2.png 以 及 p3.png， 而 在 onCreate0 时 ， 画 面 
Layout 上 的 两 个 ImageButton (图像 按钮 各 自 显示 pl.png 以 及 p2.png， 当 单 击 任 一 按钮 ， 
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则 改变 自己 的 图 片 为 p3.png， 并 还 原 另 一 按钮 为 默认 的 图 文件 ， 以 按钮 事件 及 图 文件 置换 的 


方式 来 提醒 用 户 现在 所 “选择 ”的 图 像 按钮 是 什么 。 


本 实例 实现 代码 保存 在 “光盘 :\daima\4\ practice11”， 下 面 开 始 讲解 本 实例 的 


2. 主 程序 文件 


内 体 实现 


本 实例 的 主 程序 文件 是 practicell.java， 它 通过 setImageDrawable() 方 法 来 改变 按钮 的 图 


片 , 图片 是 放 在 res/drawable 目录 下 面 。 而 ImageButton.setOnClickListener() 的 角色 则 是 本 程序 


的 关键 ,通过 同时 更 换 两 个 按钮 的 图 片 来 表示 被 选择 的 图 片 ， 是 一 种 类 似 被 选择 的 假象 手法 
常 应 用 于 切换 开关 之 用 。 文 件 examplell.java 的 具体 实现 代码 如 下 。 


package irdc. practicel 1l; 


import irdc. practicel1.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 
import android.widget.ImageButton; 
import android.widget.TextView; 


public class examplel 1 extends Activity 
{ 
TextView myTextView; 
ImageButton myImageButton 1; 
ImageButton myImageButton 2; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
/# 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


/* 以 fndViewById0 取 得 TextView 及 ImageButton 对 象 */ 
myTextView = (TextView) findViewById(R.id.myTextView); 


myImageButton 1=(ImageButton)findViewById(R.id.myImageButton 1); 


mylmageButton 2=(ImageButton)findViewById(R.id.myImageButton 2); 


jn 


/# 为 myImageButton_1 按钮 添加 OnClickListener 事件 */ 


myImageButton 1.setOnClickListener(new Button.OnClickListener() 


{ 
public void onClick(View v) 


{ 


myTextView.setText(" 你 点 击 的 是 myImageButton_1"); 


/# 单 击 myImageButton 1 按钮 时 将 myImageButton 1 图 片 


换 成 p3 图 片 % 


myImageButton 1.setImageDrawable(getResources().getDrawable( 


R.drawable.p3)); 
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入 单 击 myImageButton 1 按钮 时 将 myImageButton 2 图 片 置 换 成 p2 图 片 冯 
mylmageButton 2.setImageDrawable(getResources().getDrawable( 
R.drawable.p2)); 
} 


)); 


/* 为 myImageButton 2 按钮 添加 OnClickListener 事件 */ 
mylmageButton 2.setOnClickListener(new Button.OnClickListener() 
{ 
public void onClick(View v) 
{ 
myTextView.setText(" 你 点 击 的 是 myImageButton_2"); 
/# 单 击 myImageButton 2 按钮 时 将 myImageButton 1 图 片 置换 成 pl 图 片 */ 
mylmageButton 1.setImageDrawable(getResources().getDrawable( 
R.drawable.p1)); 
/* 单 击 myImageButton 2 按钮 时 将 myImageButton 2 图 片 置换 成 p3 图 片 */ 
mylmageButton 2.setImageDrawable(getResources().getDrawable( 
R.drawable.p3)); 


在 上 述 代 码 中 ， 除 了 在 res/drawable 放置 图 片 方式 外 ， 也 可 以 用 系统 Android 操作 系统 默 
认 的 图 片 。 如 打 电 话 、 简 短 提示 的 图 标 等 ， 只 需 修 改 main.xml 里 ImageButton 的 属性 。 


android:src="@drawable/p1" 


android:src="(@android:drawable/sym _ action call" 


上 述 代 码 中 ， 通 过 标识 “@android: ”就 表示 引用 是 android 提供 的 ， 而 不 是 自行 导入 
的 。 实 例 执行 后 的 效果 如 图 4-22 所 示 ， 当 单 击 按钮 后 会 显示 另 一 幅 图 片 。 


[CE 
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LE 加 


mm 
了 网 ep fe ey 


图 4-22 执行 效果 
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4.1 AutoCompletelextView[| DUOUUOUDODO 


在 使 用 百度 搜索 时 ， 只 需 输入 儿 个 文字 ， 就 会 显示 可 能 的 关键 字 供用 户 选 择 。 如 图 4-23 
所 示 。 


Bai 代 百度 


新 闻 网 页 贴吧 知道 MP3 图 片 视频 地 图 
Android| 汪汪 
landroid 系 统 
jandroid 论 坛 
landroid 软件 
landroid 手 机 
landroid qq 
android 论坛 
landroid 2.1 
landroid market 
landroid 手机 
|android 软 件 


关闭 | 


图 4-23 ”百度 的 搜索 提示 功能 


上 述 功能 在 Android 中 是 非常 容易 实现 的 ， 通 过 Android 的 AutoCompleteTextView (上 自 
动 提示 输入 来 源 组 件 )， 只 要 搭配 ArrayAdapter (数据 视图 ) 就 能 设计 出 类 似 百 度 搜索 提示 
的 效果 。 

1. 实例 介绍 

本 范例 先 在 Layout 当中 布局 一 个 AutoCompleteTextView Widget, 然后 通过 预先 设置 好 的 
字符 串 数 组 ， 将 此 字符 串 数组 放 入 ArrayAdapter， 最 后 利用 AutoCompleteTextView.setAdapter 
方法 , 就 可 以 让 AutoCompleteTextView 组 件 具 有 自动 完成 提示 的 功能 。 例 如 ， 只 要 输入 “ab” 
就 会 自动 带 出 包含 “ab” 的 所 有 字符 串 列表 。 

2. 主 程序 文件 

本 实例 的 主 程序 文件 是 examplel12.java， 主 要 演示 了 AutoCompleteTextView 的 用 法 。 此 
外 ， 将 ArrayAdapter 添加 AutoCompleteTextView 对 象 中 ， 所 使 用 的 方法 为 setAdapter0， 当 中 
传输 唯一 的 参数 类 型 即 为 字符 串 类 型 的 ArrayAdapter。 文 件 example12.java 的 具体 实现 代 
码 如 下 。 


package irdc.example12; 


import irdc.example12.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.widget. ArrayA dapter; 
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import android.widget.AutoCompleteTextView; 


public class example12 extends Activity 
private final String[] autoStr = new String[] 
{"a 1 人 "al 8 , "abcd", "abcde" 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/* 载 入 布局 文件 main.xml */ 


setContentView(R.layout.main); 


新 建 ArrayAdapter 对 象 ， 并 将 autoStr 字符 串 数 组 传 入 这 个 对 象 */ 
ArrayAdapter<String> adapter =new ArrayAdapter<String>(this, 
android.R.layout.simple dropdown item lline, autoStr); 


/* 以 findViewById0 方 法 取得 AutoCompleteTextView 对 象 的 内 容 */ 
AutoCompleteTextView myAutoCompleteTextView = 
(AutoCompleteTextView) findViewById(R.idmyAutoCompleteTextView); 


/ix 将 ArrayAdapter 添加 AutoCompleteTextView 对 象 中 */ 
myAutoCompleteTextView.setAdapter(adapter); 


执行 后 会 显示 一 个 输入 表单 , 当 输入 一 个 字符 时 会 显示 对 应 的 提示 词组 如 图 4-24 所 示 。 


图 4-24 输入 提示 


另外 ， 在 Android 中 有 个 类 似 AutoCompleteTextView 的 对 象 MultiAutoCompleteTextView， 
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它 继承 了 AutoCompleteTextView， 差 别 在 于 它 可 以 在 输入 框 一 直 增 加 新 的 选择 值 ， 其 编写 方 
式 也 有 些 不 同 ， 一 定 要 setTokenizer， 奋 则 会 出 现 错 误 。 


4.2 Analogcocdqj000D0DD 


Android 中 的 AnalogClock〔( 时 钟 组 件 ) 是 一 个 时 钟 对 象 ， 本 节 实 例 将 实现 一 个 时 钟 效 果 ， 
并 在 其 下 放置 一 个 TextView， 为 了 做 对 照 ， 上 面 放置 的 为 模拟 时 钟 ， 下 面 的 TextView 用 于 模 
拟 电 子 时 钟 ， 将 AnalogClock 的 时 间 以 数字 钟 形 式 显示 。 

1， 实现 原理 
在 具体 实现 上 ， 是 android.os.Handler、java.lang.Thread 以 及 android.os.Message 三 对 象 的 
整合 应 用 的 结果 , 通过 产生 Thread (线程 ) 对 象 , 在 进程 内 同步 调用 System.currentTimeMillis() 
取得 系统 时 间 ， 并 通过 Message〈 通 知 ) 对 象 来 通知 Handler( 操 作 ) 对 象 ，Handler 则 扮演 联 
系 Activity 与 Thread 之 间 的 桥梁 , 在 收 到 Message 对 象 后 , 将 时 间 变 量 的 值 , 显示 于 TextView 
当中 ， 产 生 数 字 时 钟 的 外 观 与 功能 。 

本 实例 实现 代码 保存 在 “光盘 :\daima\4\example13”， 下 面 开 始 讲解 本 实例 的 具体 实 
现 流程 。 

2. 主 程序 介绍 

本 实例 的 主 程序 文件 是 example13.java, 它 需 要 另外 加 载 Java 的 Calendar 与 Thread 对 象 ， 
并 在 onCreate0 中 构造 Handler 与 Thread 两 对 象 ， 并 实现 handelMessage0 和 run() 两 个 方法 ， 
文件 example13.java 的 具体 实现 代码 如 下 。 


package irdc.example13; 


import android.app.Activity; 

import android.os.Bundle; 

/# 这 里 我 们 需要 使 用 Handler 类 与 Message 类 来 处 理 运行 线程 */ 
import android.os.Handler; 


import android.os.Message; 

import android.widget.AnalogClock; 

import android.widget.TextView; 

必需 要 使 用 Java 的 日 历 与 线程 类 来 取得 系统 时 间 */ 
import irdc.example13.R; 


import java.util.Calendar; 
import java.lang.Thread,; 


public class example13 extends Activity 
{ 
从 声明 一 常数 作为 判别 信息 用 六 
protected static final int GUINOTIFIER = 0x1234; 


放声 明 两 个 widget 对 象 变 量 */ 
private TextView mTextView; 
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public AnalogClock mAnalogClock; 


放声 明 与 时 间 相 关 的 变量 */ 


public Calendar mCalendar; 


public int mMinutes; 
public int mHour; 


/* 声 明 关 键 Handler 与 Thread 变量 */ 
public Handler mHandler; 
private Thread mClockThread; 


/** Called when the activity is first created. */ 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 通 过 findViewById 取得 两 个 widget 对 象 */ 
mlextView=(TextView)findViewById(R.id.myTextView); 
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mAnalogClock=(AnalogClock)findViewById(R.id.myAnalogClock); 


/# 通 过 Handler 来 接收 运行 线程 所 传递 的 信息 并 更 新 TextView*/ 


mHandler = new Handler() 
{ 
public void handle Message(Message msg) 
{ 
上 * 这 里 是 处 理 信 息 的 方法 */ 
switch (msg.what) 


{ 


case example13.GUINOTIFIER: 

旋 在 这 处 理 要 TextView 对 象 Show 时 间 的 事件 */ 
mTextView.setText(mHour+" : "+mMinutes); 
break; 


} 
Super.handleMessage(msg); 


} 
人 


人 * 通 过 运行 线程 来 持续 取得 系统 时 间 * 
mClockThread=new LooperThread(); 
mClockThread .start(); 


族 改 写 一 个 线程 类 用 来 持续 取得 系统 时 间 */ 
class LooperThread extends Thread 


{ 
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public void run() 
{ 


super.run(); 


try 


{ 


do 


{ 


上 取得 系统 时 间 */ 
long time = System.current TimeMillis(); 

/# 通 过 日 历 对 象 来 取得 小 时 与 分 钟 % 

final Calendar mCalendar = Calendar.getInstance(); 
mCalendar.setTimeInMillis(time); 
mHour = mCalendar.get(Calendar. HOUR); 
mMinutes = mCalendar.get(Calendar.MINUTE); 


刻 让 运行 线程 休息 一 秒 */ 
Thread.sleep(1000); 


2 


| 


Message m = new Message(); 
m.what = example13.GUINOTIFIER; 
examplel3.this.mHandler.sendMessage(m); 


EE 要 关键 程序 :取得 时 间 后 发 出 信息 给 Handler*/ 


}while(example13.LooperThread.interrupted(O==false); 


/#* 当 系统 发 出 
} 
catch(Exception e) 
{ 
e.printStack Trace(); 
} 
} 
} 
在 上 述 代 码 


的 处 理 ， 改 为 使 


import android.widget. AnalogClock 
mDigitalClock =(DigitalClock)findViewBylId(R.id.myDigitalClock); 


本 实例 中 使 用 


FP 断 信息 时 停止 本 循环 */ 


， 要 达到 本 范例 效果 的 代码 应 该 只 有 2 行 ， 即 对 TextView 和 线程 Thread 
用 widget.DigitalClock 的 方式 ， 具 体 写法 如 下 。 


了 TextView 来 模拟 DigitalClock〈 数 字 时 钟 ) 的 做 法 ， 实 际 上 ， 也 是 参考 


AnalogClock (模拟 时 钟 ) 与 DigitalClock 这 两 个 Widget 的 程序 代码 所 做 的 练习 ， 对 于 将 来 实 


现 Timer 相关 的 小 对 象 ， 会 有 所 卉 


执行 后 会 显示 一 个 数字 时 外 


帮助 。 
和 效果。 如 图 4-25 所 示 。 


另外 ，Android 还 提供 了 System.currentTimeMillisO0、uptimeMillisO0 和 elapsedRealtimeO 这 
三 种 不 同 特性 的 System Clock 给 


准 的 Clock 


ji; 云 ， 


需要 搭配 真 


来 控制 程序 和 UI。 
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实 的 


日 期 与 时 间 使 月 


-发 者 使 用 。 本 范例 使 用 System.currentTimeMillis() 就 是 标 


日 ,为 外 两 者 则 是 适 月 


月 于 interval 与 elapse time 
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图 4-25 执行 效果 


4.B DatePicker] TimePicker]| 0D 000O0 


在 现实 中 的 许多 应 用 中 ， 经 常 需要 用 户 输入 日 期 与 时 间 格 式 的 数据 ， 很 多 站 点 给 用 户 提 
供 了 直接 在 万 年 历 上 来 选择 日 期 与 时 间 ， 选 择 完毕 后 会 自动 将 日 期 与 时 间 带 入 需要 填写 的 字 


六 


1， 实 例 介 绍 
在 Android 中 也 提供 了 类 似 的 组 件 用 于 实现 上 述 功能 。 在 本 实例 中 ， 将 演示 使 用 Android 
API 中 提供 的 DatePicker 〈 日 期 选择 组 件 ) 与 TimePicker〈 时 间 选 择 组 件 ) 两 种 对 象 来 实现 动 
态 输入 日 期 与 时 间 的 功能 。 实 例 中 使 用 了 DatePicker、TimePicker 与 TextView 三 种 对 象 ， 以 
TextView 来 显示 日 期 与 时 间 ， 默 认 带 入 目前 系统 的 日 期 与 时 间 ，DatePicker 与 TimePicker 
可 让 用 户 动 态 调整 日 期 与 时 间 ， 当 用 户 调整 了 DatePicker 的 日 期 或 TimePicker 时 间 时 ， 则 
TextView 中 所 显示 的 日 期 与 时 间 亦 会 跟着 改变 。 

本 实例 实现 代码 保存 在 “光盘 :daimav4\example14” 下面 开 始 讲 解 本 实例 的 具体 实现 


2. 主 程序 介绍 

本 实例 的 主 程序 文件 是 examplel4.java, 以 updateDisplay() 这 个 方法 来 设置 TextView 中 所 
显示 的 日 期 时 间 ， 以 java.util.Calendar 对 象 来 取得 目前 的 系统 时 间 ， 并 预先 带 入 TextView 中 。 
当 用 户 更 改 了 DatePicker 里 的 年 、 月 、 日 时 ， 将 触发 DatePicker 的 onDateChange() 事件 ， 运 
行 updateDisplay() 来 重新 设置 TextView 中 显示 的 日 期 ; 同样 的 原理 , 当 用 户 更 改 了 TimePicker 
里 的 时 间 ， 会 触发 TimePicker 的 onTimeChange0 事件 ， 运 行 updateDisplay() 来 重新 设置 
TextView 中 显示 的 时 间 。 

文件 example14.java 的 具体 实现 代码 如 下 。 


package irdc.example14; 


庆 加 载 相关 类 */ 
import irdc.example14.R; 


import java.util.Calendar; 
import android.app.Activity; 
import android.os.Bundle; 
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import android.widget.TextView; 


import android.widget.DatePicker; 
import android.widget.TimePicker; 


public class example14 extends Activity 


{ 
放声 明日 期 及 时 间 变 量 */ 


private int mY ear; 


private int mMonth; 
private int mDay; 
private int mHour; 
private int mMinute; 
放声 明 对 象 变 量 */ 
TextView tv; 
TimePicker tp; 
DatePicker dp; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
族 取 得 目前 日 期 与 时 间 *%/ 
Calendar c=Calendar.getInstance(); 
mYear=c.get(Calendar.YEAR); 
mMonth=c.get(Calendar.MONTH); 
mDay=c.get(Calendar.DAY OF MONTH); 
mHour=c.get(Calendar.HOUR OF DAY); 
mMinute=c.get(Calendar.MINUTE); 


super.onCreate(savedInstanceState); 
/# 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


/# 取 得 TextView 对 象 ， 并 调用 更 新 显示 方法 updateDisplay() 
来 设置 显示 的 初始 日 期 时 间 */ 

tv= (TextView) findViewById(R.id.showTime); 

updateDisplay(); 


/# 取 得 DatePicker 对 象 ， 以 init0 

设置 初始 值 ， 通 过 onDateChangeListener0 监 听 时 间 变 化 */ 
dp=(DatePicker)findViewById(R.id.dPicker); 
dp.init(mYear,m Month,mDay,new DatePicker.OnDateChangedListener() 
{ 

@Override 

public void onDateChanged(DatePicker view,int year, 

int monthOfY ear,int dayOf Month) 
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mY ear=year; 

mMonth= monthOfY ear; 

mDay=dayOfMonth; 

/* 调 用 updateDisplay0 来 改变 显示 日 期 */ 

updateDisplay(); 

} 

)); 
人 # 取 得 TimePicker 对 象 ， 并 设置 为 24 小 时 制 显 示 */ 
tp=(TimePicker)findViewBylId(R.id.tPicker); 
tp.setIs24HourView(true); 


/*setOnTimeChangedListener， 并 和 窗 新 onTimeChanged event*/ 
tp.setOnTimeChangedListener(new TimePicker.OnTimeChangedListener() 


{ 
(QOverride 
public void onTimeChanged(TimePicker view, 
int hourOfDay, 
int minute) 
{ 
mHour=hourOfDay; 
mMinute=minute; 
/# 调 用 updateDisplay0 来 改变 显示 时 间 */ 
updateDisplay(); 
} 
)); 


旋 设 置 显示 日 期 时 间 的 方法 */ 
private void updateDisplay() 
{ 
tv.setText( 
new StringBuilder().append(mYear).append("/") 
.append(format(m Month + 1)).append("/") 
.append(format(mDay)).append(" ") 


.append(format(mHour)).append(":") 
.append(format(mMinute)) 


访 日 期 时 间 显 示 两 位 数 的 方法 */ 
private String format(int x) 


{ 
String s=""+x; 
ifls.length()==1) s="0"+s; 
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return s; 


} 
} 


执行 后 会 显示 一 个 数字 时 钟 效果 。 用 户 可 以 分 别 单 击 十 和 一 来 自动 选择 日 期 和 时 间 。 
具体 效果 如 图 4-26 所 示 。 
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图 4-26 ”执行 效果 


4. ProgressBarl| Handlerl 0UDODDOD 


ProgressBar 是 一 个 进度 条 组 件 ， 能 够 实现 网 络 中 的 进度 条 效果 。 如 图 4-27 所 示 。 


ProgressBarTest 


图 4-27 进度 条 


进度 条 分 不 确定 和 确定 2 种 。 默 认 值 是 不 确定 (indeterminate=true )。 在 图 4-27 中 的 第 
1-3 个 是 不 确定 进度 条 ， 第 4 个 是 确定 进度 条 。 

进度 条 有 4 种 风格 可 以 使 用 。 

口 默认 值 是 progressBarStyle， 图 4-27 中 的 的 第 2 个 

口 设置 成 progressBarStyleSmall 后 ， 图 标 变 小 。 图 4-27 中 的 第 1 个 。 
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访问 , 并 将 运 
最 后 由 Activity 的 Handler 习 
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人 


口 设置 成 progressBarStyleLarge 后 ， 图 标 变 大 。 图 4-27 中 的 第 3 个 。 
口 设置 成 progressBarStyleHorizontal 后 ， 变 成 横向 长 方形 。 图 4-27 中 的 第 4 个 。 


五 


请 实 例 中 ， 使 用 ProgressBar 〈 进 度 条 组 件 ) 和 Handler 实现 了 后 台 进 度 提示 效果 。 
ProgressBar 实现 了 进度 条 ， 而 且 利 用 Handler 实现 了 新 进程 对 Activity 中 Widget 的 
云 行 状态 显示 出 来 。 通 过 Handler 对 象 和 Message 对 象 , 将 进程 里 的 状态 往外 传递 ， 
来 取得 运行 状态 。 
例 实现 代码 保存 在 “光盘 :daimav4vexample1$”， 主 程序 文件 是 example15.java， 具 体 


二 
个 


实现 代码 如 下 。 


package irdc.example15; 


import irdc.example15.R; 

import android.app.Activity; 
import android.os.Bundle; 

import android.os.Handler; 

import android.os.Message; 

import android.view.View; 

import android.widget.Button; 
import android.widget.ProgressBar; 
import android.widget.TextView; 


public class example15 extends Activity 
{ 
private TextView mTextView01; 
private Button mButton01; 
private ProgressBar mProgresSBar01; 
public int intCounter=0; 


旋 自 定义 Handler 信息 代码 ， 用 以 作为 识别 事件 处 理 */ 
protected static final int GUI STOP NOTIFIER = 0x108; 
protected static final int GUI THREADING NOTIFIER = 0x109; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButton01 = (Button)findViewById(R.id.myButton!1); 
mTextView01 = (TextView)findViewByld(R.id.myTextView!); 


庶 设置 进度 条 对 象 六 
mpProgressBar01 = (ProgressSBar)findViewById(R.id.myProgresSBarl); 
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/* 调用 setIndeterminate 方法 赋值 indeterminate 模式 为 false */ 
mProgressBar01.setIndeterminate(false); 


旋 当 单 击 按钮 后 ， 开 始 运行 线程 工作 * 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 

{ 


族 单 击 按钮 让 进度 条 显示 党 
mTextView01.setText(R.string.str_ progress_ start); 


放 将 隐藏 的 进度 条 显示 出 来 刀 
mpProgressBar01.setVisibility(View.VISIBLE); 


/* 指定 mProgressBar01 对 象 为 最 多 100 */ 
ImProgresSBar01.setMax(100); 


/* 初始 化 mProgressBar01 对 象 为 0 */ 
mProgressBar01.setProgress(0); 


上 # 起 始 一 个 运行 线程 */ 
new Thread(new Runnable() 
{ 
public void run() 
{ 
旋 默认 0 至 9， 共 运行 10 次 的 循环 叙述 */ 
for (int 1=0;i<10;i++) 
{ 
try 
{ 
庆 成 员 变 量 ， 用 以 识别 加 载 进度 * 
intCounter = (i+1)*20; 
此 每 运行 一 次 循环 ， 即 暂停 1 秒 */ 
Thread.sleep(1000); 


谨 当 Thread 运行 5 秒 后 显示 运行 结束 */ 
if(i==4) 
{ 
/# 以 Message 对 象 ， 传 递 参 数 给 Handler */ 
Message m= new Message(); 


放 以 what 属性 指定 User 自 定义 *%/ 
m.what = examplel5.GUI STOP_ NOTIFIER; 
examplel$.this.myMessageHandler.sendMessage(m); 
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break; 
} 


else 


{ 


Message m= new Message(); 
m.what = examplel5.GUI THREADING NOTIFIER; 
examplel$.this.myMessageHandler.sendMessage(m); 


} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 
}).startO; 


/* Handler 建构 之 后 ， 会 聆听 传 来 的 信息 汉 
Handler myMessageHandler = new Handler() 
{ 
/* (Override */ 
public void handleMessage(Message msg) 
{ 
switch (msg.what) 
{ 
旋 当 取 得 识别 为 离开 运行 线程 时 所 取得 的 信息 */ 
case examplel5.GUI STOP_ NOTIFIER: 


上 # 显示 运行 结束 */ 
mTextView01.setText(R.string.str progress done); 


/* 设置 ProgressBar 组 件 为 隐藏 状态 * 
mpProgressBar01.setVisibility(View.GONE); 
Thread.currentThread().interrupt(); 

break; 


/# 当 取 得 识别 为 持续 在 运行 线程 当中 时 所 取得 的 信息 */ 
case examplel5.GUI THREADING NOTIFIER: 
if(!Thread.currentThread().isInterrupted()) 
1 


mpProgressBar01.setProgress(intCounter); 
入 将 显示 进度 显示 在 文本 框 当中 */ 
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mTextView01.setText 


( 


getResources().getText(R.string.str progress start)+ 
"("+Integer.toString(intCounter)+"%)\n"+ 


"Progress:"+ 


Integer.toString(mProgressBar01.getProgress())+ 
"nn"+"Indeterminate:"+ 
Boolean.toString(mProgressBar01 .isIndeterminate()) 


六 
} 
break; 


} 


Super.handleMessage(msg); 


} 
局 
} 


在 本 实例 中 ， 设 计 了 一 个 按钮 ， 此 按钮 将 部 署 在 main.xml 中 的 进度 条 控件 显示 出 来 。 在 


默认 的 main.xml 中 没有 指定 它 的 indeterm-inate 属性 ， 所 以 即使 在 程序 中 调用 了 ProgressBar 
的 setindeterminate() 方 法 , 无 法 改变 ProgressBar.getProgress 的 值 ， 此 值 永远 是 0. 所 以 在 此 使 用 


了 循环 动画 来 当 作 运行 过 程 中 的 显示 素材 ， 并 使 用 了 一 个 counter 整数 来 递增 ， 表示 运行 的 百 


分 比 。 


执行 后 会 显示 一 个 按钮 界面 ， 月 
果 。 有 具体 效果 如 图 4-28 所 示 。 


5554: aa 


Indeterminate:true 


按 下 后 开始 运行 


昌 户 单 避 


fF“ 按 下 后 开始 运行 ”按钮 后 会 显示 进度 条 提示 效 


图 4-28 执行 效果 


4D 0000000AnrayAdapteaUDUODOO0D 


众所周知 ，GridView 是 一 个 网 格 化 的 二 维 排版 配置 视图 。 在 本 节 实 例 中 ， 将 演示 如 何 使 


用 网 格 视图 控件 来 实现 动态 排版 处 理 。 在 具体 实现 上 ， 插 入 了 2 个 按钮 ， 作 为 动态 放 入 网 格 


视图 组 件 的 开关 。 一 个 按钮 设置 GridView 为 两 列 显 示 样 式 ， 放 入 4 个 em; 另 一 个 按钮 指定 
为 3 列 显示 样式 ， 放 入 9 个 Item (条目)， 这 样 就 实现 了 对 文字 的 动态 排版 处 理 。 
本 实例 实现 代码 保存 在 “光盘 :\daimaM\example16”， 主 程序 文件 是 example16.java， 具 体 
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实现 代码 如 下 。 


package irdc.example16; 


import irdc.example16.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.view.View; 

import android.widget.AdapterView; 
import android.widget.ArrayA dapter; 
import android.widget.Button; 
import android.widget.GridView; 
import android.widget.TextView; 


public class example16 extends Activity 

{ 
private TextView mTextView0!1; 
private Button mButton01,mButton02; 
private GridView mGridView01; 
private String[] mGamesl,mGames2; 
private ArrayAdapter<String> aryA dapterl; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


从 4 个 字符 串 数组 六 
mGamesl = new String[] 
{ 
getResources().getString(R.string.str_list1), 
getResources().getString(R.string.str_ list2), 
getResources().getString(R.string.str_list3), 
getResources().getString(R.string.str list4) 


据 


入 9 个 字符 串 数 组 冯 
mGames2 = new String[] 
{ 

getResources().getString(R.string.str_list1), 
getResources().getString(R.string.str list2), 
getResources().getString(R.string.str_list3), 
getResources().getString(R.string.str_list4), 
getResources().getString(R.string.str_list1), 
getResources().getString(R.string.str_list2), 
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getResources(.getString(R.string.str_ list3)， 
getResources().getString(R.string.str_list4), 
getResources().getString(R.string.str_list1) 


六 


mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton02 = (Button)findViewById(R.id.myButton2); 
mGridView01 = (GridView)findViewByld(R.id.myGridView!); 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 


mButton01.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 

{ 


人/#4 个 元 素 ， 以 2 列 方式 呈现 (2x2) */ 
mGridView01.setNumColumns(2); 


aryAdapterl = new ArrayAdapter<String> 
(example16.this, R.layout.simple list item 1 small, mGames1l); 


mGridView01.setAdapter(aryAdapter!); 
mGridView01.setScrollBarStyle(DEFAULT KEYS DIALER); 
mGridView01.setSelection(2); 
mGridView01.refreshDrawableState(); 
} 
)); 


mButton02.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
public void onClick(View v) 


{ 


/*9 个 元 素 ， 以 3 列 方式 呈现 (3x3) */ 
mGridView01.setNumColumns(3); 


aryAdapterl = new ArrayAdapter<String> 
(example16.this, R.layout.simple list item 1 small, mGames2); 


mGridView01.SetAdapter(aryAdapter1); 
} 
D); 


mGridView01.setOnItemClickListener 
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(new GridView.OnIltemClickListener() 


{ 
(QOverride 


public void onItemClick(AdapterView<?> parent, 
View v int position, long arg3) 


{ 


从 判断 Adapter 里 的 元 素 个 数 ， 判 断 被 点 击 的 是 第 儿 个 元 素 名 称 */ 
switch(aryAdapterl.getCount()) 


{ 


case 4: 


/* position 为 GridView 日 


的 元 素 索引 值 */ 


ImTextView01.setText(mGames1[position]); 


break; 
case 9: 


mTextView01.setText(mGames2[position)); 


break; 


在 实例 中 ， 设 置 了 mButton01 和 mButton02 两 个 按钮 ， 设 置 各 自 OnClickListener( 单 击 


事件 )， 在 按钮 事件 中 处 理 要 配置 GridView 对 象 的 方法 。GridView 的 setNumColumns 方法 为 
设置 其 字段 数量 ， 为 了 方便 配置 GridView 对 象 。 通 过 使 用 ArrarAdapter,，Android.widget. 


ArrarAdapter 默认 构造 需要 3 个 参数 ， 并 将 初始 化 的 ArrarAdapter 对 象 ArrarAdapterl 以 调用 


GridView.setAdapter0 的 方式 ， 将 String 类 型 的 Item 放 入 到 GridView 对 象 中 去 。 
执行 后 会 显示 2 个 按钮 的 界面 ， 当 单 击 “ 显 示 4 个 ”按钮 后 会 显 2X2 排列 的 样式 ， 
效果 如 图 4-39 所 示 ; 当 单 击 “ 显 示 9 个 ”按钮 后 会 显 3X3 排列 的 样式 ， 效 果 如 图 4-40 


所 示 。 
se 支 天 全 6:46 Am 
lexample16 四 9 
实现 动态 排版 处 理 
AAAAA BBBBB 
CCCCC DDDDD 


显示 4 个 (2x2) | | 显示 9 个 (3x3) 


图 4-29 2X2 排列 


6:46 am 


[Example16 
实现 动态 排版 处 理 


AAAAA BBBBB CCCCC 
DDDDD AAAAA BBBBB 
CCCCC DDDDD AAAAA 


显示 4 个 (2x2) | | 显示 9 个 (3x3) 


图 4-30 3X3 排列 
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4.]J [D0 LstAdivty 


ListActivity (列表 视图 组 件 ) 也 是 一 个 独立 的 类 ， 它 和 Activity 是 同一 级 别 的 类 。 
ListActivity 也 能 够 实现 布局 处 理 ， 能 用 于 显示 菜单 列表 、 列 表明 细 项 目 。 在 本 节 的 内 容 中 ， 
将 详细 介绍 ListActivity 的 基本 知识 。 


4.16.1 ListActivity 介 绍 
在 android.app 包 里 包含 了 几 个 重要 的 类 ，ListActivity 就 是 其 中 之 一 。ListActivity 类 
其 实 就 是 一 个 含有 一 个 ListView 组 件 的 Activity 类 。 也 就 是 说 ， 如 果 直 接 在 一 个 普通 的 
Activity 中 自己 加 一 个 ListView， 也 是 完全 可 以 取代 这 个 ListActivity 的 ， 只 是 它 更 方便 
而 已 。 

ListActivity 像 数组 或 者 是 光标 一 样 ， 将 绑 定 的 数据 资源 以 陈列 选项 的 方式 显示 内 容 。 当 
读者 选择 这 些 选项 时 ， 将 会 触发 一 个 事件 。ListActivity 主持 操作 着 一 个 列表 视图 对 象 ， 这 个 
列表 视图 能 绑 定 不 同 的 数据 资源 ， 典 型 的 就 是 一 个 持 有 查询 结果 的 数组 或 者 是 光标 。 

1. 屏幕 布局 (Screen Layout) 

ListActivity 有 一 个 默认 的 布局 ， 这 个 布局 由 单一 的 、 全 屏 列 表 构 成 。 通 过 在 onCreate() 
方法 中 使 用 setContentViewO 设 置 自己 的 视图 布局 来 定制 屏幕 布局 。 如 果 要 完成 这 些 ， 我 们 上 自 
己 的 视图 必须 包含 一 个 id 为 “@android:id/list” 的 ListView 对 象 。 如 果 列 表 为 空 时 ， 可 以 包 
含 男 外 一 个 视图 对 象 ， 这 个 空 的 列表 必须 有 一 个 “android:empty” 值 的 id， 注 意 到 当 有 一 个 
空 的 视图 显示 时 ， 这 个 列表 视图 将 会 在 没有 任何 数据 时 被 隐藏 。 

2. 排 布局 (Row Layout) 

可 以 在 列表 中 确定 单一 的 排 ， 它 是 通过 在 活动 所 操作 的 ListAdapter 对 象 中 设 定 的 一 个 资 
源 布局 来 实现 的 。 一 个 ListAdapter 持 有 一 个 的 参数 来 为 每 一 行 确定 了 所 使 用 到 的 布局 的 资 
源 。 同 时 ， 它 还 有 另外 两 个 参数 ， 这 两 个 参数 让 你 明确 与 哪个 对 象 相互 关 联 ， 通 常 是 两 个 平 
行 的 数组 。Android 提供 了 一 些 标准 的 行 布局 资源 。 这 些 都 是 在 Rayout 类 中 所 定义 的 ， 名 称 


Rs 


像 simple list item 1，simple list item 2，two_line list_ item 之 类 一 样 。 
下 面 先 来 分 析 一 段 代 码 ， 这 段 代码 就 是 使 用 了 这 个 方面 的 知识 。 
String Songss[]; 
/首先 是 创建 一 个 File 类 型 的 实例 对 象 home 
File home = new File(MEDIA PATHD; 
/ 列 出 目录 文件 中 的 所 有 文件 ， 并 将 这 些 文件 名 称 放 到 字符 串 数组 中 
songss=home.list(); 
/下 面 是 关键 代码 ， 创 建 一 个 ArrayAdapter 对 象 ， 并 将 三 个 参数 分 别 置 为 this( 表 明 是 当前 上 下 文 ), 
//R.layout.song item 表明 要 在 song item 中 显示 ,songss( 这 就 是 显示 在 song item 中 的 内 容 ), 这 样 就 
将 要 显示 在 ListView 中 的 内 容 设置 好 了 。 
ArrayAdapter<String>songList = new ArrayAdapter<String> (this,R.layout.song item,songss); 
/使 用 setListAdapter0， 就 是 使 ArrayAdapter 类 型 的 songList 运行 ， 即 真正 起 作用 
setListAdapter(songList); 


通过 上 述 代码 介绍 了 知识 点 ListView 中 ListAdapter 的 使 用 。 
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同时 ，Android 不 仅仅 提供 了 数组 ， 还 提供 了 另外 两 个 标准 的 列表 适配器 : 处 理 static 数 
据 类 型 的 SimpleAdapter 和 处 理光 标 结果 查询 的 SimpleCursorAdapter。 人 例如， 下面 是 一 个 将 名 
称 和 公司 信息 绑 定 到 一 个 两 行 排 布局 的 活动 列表 视图 〈 所 谓 的 两 行 排 指 的 是 两 行 数值 作 一 个 
排 布局 ， 也 可 以 是 三 行 作为 一 个 排 布局 或 者 是 一 行 作为 一 个 行 布局 )。 


public class MyListAdapter extends ListActivity { 
(QOverride 
protected void onCreate(Bundle icicle){ 
super.onCreate(icicle); 
setContentView(R.layout.custom list activity view); 
mCursor = People.query(this.getContentResolver(), null); 
startManagingCursor(mCursor); 
ListAdapter adapter = new SimpleCursorAdapter( 
this, 
android.R.layout.two _ line list_item, 
mCursor, 
new String[] {People.NAME, People.COMPANY }， 
new int[]); 
SetListAdapter(adapter); 


} 


3. 可 实现 的 方法 
如 果 让 程序 继承 ListActivity， 则 可 以 实现 下 面 的 方法 。 
口 getListAdapter0: 获取 列表 项 目的 Adapter。 

口 getListView: 获取 列表 的 View。 

口 getSelectItemId0: 获取 当前 Keypad 所 选择 的 Item ID 。 
口 

口 


onContentChanged(): ListActivity 列表 内 容 更 新 事件 。 
onListItemClick(ListView,View,int,long): User 在 列表 项 目 中 单 击 触发 事件 。 
口 onRestoreInstancsState(Bundle): 还 原 至 此 实例 状态 事件 。 
口 setListAdapter(ListAdapten: 设置 ListAdapter 的 列表 项 目 。 
口 setSelection(int): 设置 所 选 的 项 目 。 
在 使 用 ListActivity 时 ， 并 不 用 像 使 用 Activity 那样 必须 使 用 setContentView 方法 来 设置 
版 型 Layout 才能 显示 页 面 。ListActivity 在 不 需要 重 写 protected void onCreate(Bundle 
savedInstanceState) 的 情况 下 ， 直 接 将 列表 加 载 到 ListActivity 中 ， 这 样 使 用 起 来 十 分 方便 ， 经 
常用 于 投票 选择 和 多 项 目 列表 条 中 去 。 


4.16.2 ”Listactivity 应 用 方法 


在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 来 讲解 使 用 ListActivity 的 具体 流程 。 本 
实例 实现 代码 保存 在 “光盘 :daimaexample17”， 是 通过 Listactivity 和 Menu 联合 实现 的 。 
本 实例 的 具体 实现 流程 如 下 。 
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打开 eclipse， 依 次 单 击 “File” 一 “New” 一 “Android Project”， 


“example17” 的 工程 文件 。 
第 二 步 : 编写 布局 文件 main.xml 实现 整体 布 


可 


<?xml version="1.0" encodine="utf-8"?> 

<TextView 
xmlns:android="http://schemas.android.com/apk/res/android" 
android:id="@+Hid/myTextView1" 
android:layout_width="fill parent" 
android:layout_ height="wrap_content" 
android:text="(@string/hello" 

/> 


第 三 步 : 在 string.xml 添加 程序 中 要 使 用 的 字符 串 ， 有 具体 代码 


<?xml version="1.0" encoding="utf-8"?> 
<resources> 
<string name="hello"> 你 好 雨夜 </string> 
<string name="app_name"> 欢 迎 你 </string> 
<string name="str_list1"> 精 绝 古 城 </string> 
<string name='"str_list2"> 龙 岭 迷 窟 </string> 
<string name="str list3"> 云 南 虫 谷 </string> 


<string name="str_list4"> 昆 仓 神 宫 </string> 


<string name="str_list5">C++ 语 言 </string> 

<string name="str_list6">Java 语言 </string> 

<string name="str_list7">PHP 语言 </string> 

<string name="str_list8">Basic 语言 </string> 
<string name="str menu listl"> 显 示 列 表 1</string> 
<string name="str menu list2"> 显 示 列 表 2</string> 


</resources> 


L 体 代码 如 下 。 


如 下 


第 四 步 : 编写 example17.java， 实 现 显 示 功 能 ， 其 具体 实现 流程 如 下 。 


声明 对 象 变量 。 
2) 载 入 main.xml Layout。 
3) 初始 化 对 象 ， 并 将 mButton 添加 到 onClickListener 〈 监 听 自 


击 事件 )。 


4) 取得 输入 的 关键 字 。 
5) 根据 搜索 文件 的 method 和 关键 字 搜 索 。 
文件 shiyongListActivityjava 的 具体 实现 代码 如 下 : 


package irdc.example17; 


import irdc.example17.R; 
import android.app.ListActivity; 
import android.os.Bundle; 


128 国 面 


新 建 一 个 名 为 


第 4 章 


import android.view.Menu; 

import android.view.Menultem; 
import android.view.View; 

import android.widget. ArrayA dapter; 
import android.widget.ListView; 


import android.widget.Toast; 


public class example17 extends ListActivity 

{ 
private int selectedltem = —1; 
private String[] mString; 
static final private int MENU_LIST1 = Menu.FIRST; 
static final private int MENU_LIST2 = Menu.FIRST+1; 
private ArrayAdapter<String> mla; 


(@Override 


protected void onCreate(Bundle savedInstanceState) 


{ 
/TODO 自动 引起 的 方法 残余 部 分 */ 


super.onCreate(savedInstanceState); 


(QOverride 
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protected void onListItemClick(ListView 1, View v, int position, long id) 


{ 
/TODO 自动 引起 的 方法 残余 部 分 */ 


selectedItem = position; 


Toast.makeText(shiyongListActivity.this, mString[selectedIteml], Toast.LENGTH SHORT).show(); 


super.onListItemClick(], v, position, id); 


(@Override 
public boolean onCreateOptionsMenu(Menu menu) 
{ 

/menu 组 ID*/ 

int idGroup1l = 0; 


旋 项 目的 顺序 位 置 */ 
int orderMenuIteml = Menu.NONE; 
int orderMenuItem2 = Menu.NONE+1; 


menu.add(idGroupl1, MENU LIST1, orderMenultem!1, R.string.str menu list1l); 
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menu.add(idGroupl1, MENU LIST2, orderMenuItem2, R.string.str menu list2); 


return super.onCreateOptionsMenu(menu); 


} 
@Override 
public boolean onOptionsItemSelected(Menultem item) 
{ 
switch(item.getltemld()) 
{ 
case (MENU LIST1): 
mString = new String[] 
{ 
getResources().getString(R.string.str_list1), 
getResources().getString(R.string.str_list2), 
getResources().getString(R.string.str_list3), 
getResources().getString(R.string.str_list4) 
上 
mla= new ArrayAdapter<String>(shiyongListActivity.this, R.layout.main, mString); 
shiyongListActivity.this.setListAdapter(mla); 
break; 
case (MENU LIST2): 
mString = new String[] 
{ 
getResources().getString(R.string.str list$), 
getResources().getString(R.string.str_list6), 
getResources().getString(R.string.str_list7), 
getResources().getString(R.string.str_list8) 
} 
mla= new ArrayAdapter<String>(shiyongListActivity.this, R.layout.main, mString); 
shiyongListActivity.this.setListAdapter(mla); 
break; 
} 
return super.onOptionsItemSelected(item); 
} 
} 


执行 后 将 在 底部 显示 2 个 布局 块 ， 并 分 别 显示 设置 的 文本 ， 具 体 效果 如 图 4-31 所 示 ; 当 
单 击 第 一 个 布局 块 后 会 显示 弹出 对 应 的 信息 ， 如 图 4-32 所 示 ; 当 单 击 第 二 个 布局 块 后 会 显示 
弹出 对 应 的 信息 ， 如 图 4-33 所 示 。 
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图 4-31 初始 效果 图 4-32 ”对 应 的 信息 图 4-33 ”对 应 的 信息 
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在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 来 讲解 使 用 Matrix〈 颜 色 矩 阵 对 象 ) 实 
现 图 片 缩放 的 具体 流程 。 本 实例 保存 在 “光盘 :daima\example18” 中 ， 其 具体 实现 流程 如 下 。 
第 一 步 : 打开 eclipse， 依 次 单 击 “File” 一 “New” 一 “Android Project”， 新 建 一 个 名 为 
“example18” 的 工程 文件 。 
第 二 步 : 编写 布局 文件 main.xml 实现 整体 布局 。 

通过 上 述 步骤 ， 在 页 面 中 插入 了 一 幅 图 片 ， 名 称 为 suofang.bng，2 个 Button 按钮 。 
第 三 步 : 在 string.xml 添加 程序 中 要 使 用 的 字符 串 ， 有 具体 代码 如 下 。 


<?xml version="1.0" encoding="utf-8"?> 
<resources> 
<string name="hello">Hello</string> 
<string name="app_name"> 了 雨夜 </string> 
<string name="str_buttonl"> 缩 小 处 理 </string> 
<string name="str_button2"> 放 大 处 理 </string> 


</resources> 


第 四 步 : 编写 处 理 文件 examplel18.java， 其 具体 实现 流程 如 下 。 

1) 载 入 布局 文件 main.xml Layout。 

2) 取得 屏幕 分 辩 率 大 小 ， 初 始 化 相关 变量 。 

3) 实现 缩小 按钮 onClickListener 处 理 mButton01.setOnClickListener。 
4) 实现 放大 按钮 onClickListener 处 理 mButton02.setOnClickListener。 
5) 定义 图 片 缩小 方法 small()。 

6) 定义 图 片 放 大 方法 big0。 

文件 suofang.java 的 具体 实现 代码 如 下 : 
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package irdc.example18; 


入 加 载 相关 类 */ 

import irdc.example18.R; 

import android.app.Activity; 

import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.graphics.Matrix; 

import android.os.Bundle; 

import android.util.DisplayMetrics; 
import android.view.View; 

import android.widget.AbsoluteLayout; 
import android.widget.Button; 

import android.widget.ImageView; 


public class example18 extends Activity 


{ 
/* 相关 变量 声明 */ 


private ImageView mImageView; 


private Button mButton01; 
private Button mButton02; 
private AbsoluteLayout layoutl; 
private Bitmap bmp; 

private int 1d=0; 

private int displayWidth; 
private int displayHeight; 
private float scaleWidth=1; 
private float scaleHeight=1; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
/+ 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


入 取得 屏幕 分 辨 率 大 小 */ 
DisplayMetrics dm=new DisplayMetrics(); 


getWindowManager().getDefaultDisplay().get Metrics(dm); 
displayWidth=dm.widthPixels; 

旋 屏幕 高 度 须 扣除 下 方 Button 高 度 沁 
displayHeight=dm.heightPixels-80; 

族 初始 化 相关 变量 %% 


bmp=BitmapFactory.decodeResource(getResources(), 


R.drawable.suofang); 
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mlmageView = (ImageView)findViewById(R.id.myImageView); 
layoutl = (AbsoluteLayout)findViewByld(R.id.layout!); 
mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton02 = (Button)findViewById(R.id.myButton2); 


/* 缩小 按钮 onClickListener */ 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 

(@Override 

public void onClick(View v) 

{ 

small(); 

} 

)); 


/* 放大 按钮 onClickListener */ 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 
(@Override 
public void onClick(View v) 
{ 
bigO; 


旋 图 片 缩小 的 method */ 
private void small() 
{ 
int bmp Width=bmp.getWidth(); 
int bmpHeight=bmp.getHeight(); 
民 设置 图 片 缩小 的 比例 */ 
double scale=0.8; 
入 计算 出 这 次 要 缩小 的 比例 *W 
ScaleWidth=(float) (scaleWidth*scale); 
scaleHeight=(float) (scaleHeight*scale); 


/六 产生 reSize 后 的 Bitmap 对 象 */ 

Matrix matrix = new Matrix(); 

matrix.postScale(scaleWidth, scaleHeight); 

Bitmap resizeBmp = Bitmap.createBitmap(bmp,0,0,bmp Width, 
bmpHeight,matrix,true); 


这 id--0) 
{ 
如 果 是 第 一 次 按 ， 就 删除 原来 默认 的 ImageView */ 
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layoutl.removeView(mImageView); 
} 
else 
{ 
旋 如 果 不 是 第 一 次 按 ， 就 删除 上 次 放大 缩小 所 产 生 的 ImageView */ 
layoutl.removeView((ImageView)findViewById(id)); 


} 
诺 产生 新 的 ImageView， 放 入 新 大 小 的 Bitmap 对 象 ， 再 放 入 界面 中 */ 


id++; 


ImageView imageView = new ImageView(suofang.this); 
imageView.setld(id); 
imageView.setImageBitmap(resizeBmp); 
layoutl.addView(imageView); 

SetContentView(layout1 ); 


/* 因为 图 片 放 到 最 大 时 放大 按钮 会 disable， 所 以 在 缩小 时 把 他 重 设 为 enable 
mButton02.setEnabled(true); 


大 图 后 大 大 辣 友 洪 光 
private void big() 
{ 
int bmp Width=bmp.get Width(); 
int bmpHeight=bmp.getHeight(); 
放 设置 图 片 放 大 的 比例 */ 
double scale=1.25; 
旋 计算 这 次 要 放大 的 比例 六 
scaleWidth=(float)(scaleWidth*scale); 
scaleHeight=(float)(scaleHeight*scale); 


/六 产生 新 大 小 后 的 Bitmap 对 象 */ 


Matrix matrix = new Matrix(); 


matrix.postScale(scaleWidth, scaleHeight); 
Bitmap resizeBmp = Bitmap.createBitmap(bmp,0,0,bmp Width, 
bmpHeight,matrix,true); 


if(id==0) 

{ 
旋 如 果 是 第 一 次 按 ， 就 删除 原来 设置 f 
layoutl.removeView(mImageView); 

} 

else 

{ 
旋 如 果 不 是 第 一 次 按 ， 就 删除 上 次 放大 缩小 所 产生 的 ImageView */ 
layoutl.removeView((ImageView)findViewById(id)); 


4 


图 像 */ 
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翌 产生 新 的 ImageView， 放 入 新 大 小 的 Bitmap 对 象 ， 再 放 入 Layout 中 */ 
id++; 


ImageView imageView = new ImageView(suofang.this); 
imageView.setld(id); 
imageView.setImageBitmap(resizeBmp); 
layoutl.addView(imageView); 

SetContentView(layout1 ); 


入 如 果 再 放大 会 超过 屏幕 大 小 ， 就 把 Button disable */ 

if(scaleWidth*scale*bmpWidth>displayWidth|| 
scaleHeight*scale*bmpHeight>displayHeight) 

{ 
mButton02.setEnabled(false); 

} 

} 
} 


执行 后 将 显示 一 幅 图 片 和 两 个 按钮 ， 如 图 4-34 所 示 ; 分 别 单 击 “ 缩 小 处 理 ” 和 “放大 处 
里 ”按钮 后 ， 会 实现 对 图 片 的 缩小 、 放 大 处 理 ， 如 图 4-35 所 示 。 


YH 


六 症 名 7:00 An 咏 天 后 7:01am 


图 4-34 初始 效果 图 4-35 缩小 后 效果 
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在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 过 程 ， 来 讲解 使 用 Bitmap 对 象 和 Matrix 
对 象 实现 图 片 旋转 的 具体 流程 。 本 实例 保存 在 “光盘 :\daimaw\example19” 中 ， 其 具体 实现 流 
程 如 下 。 
第 一 步 : 打开 eclipse， 依 次 单 击 “File” 一 “New” 一 “Android Project”， 新 建 一 个 名 为 
“example19” 的 工程 文件 。 
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第 二 步 : 编写 布局 文件 main.xml 实现 整体 布局 。 

通过 上 述 步 又 ， 在 页 面 中 搬入 了 一 幅 图 片 ， 名 称 为 hippo.png，2 个 Button 按钮 ，1 个 
TextView 控件 。 
第 三 步 : 在 string.xml 添加 程序 中 要 使 用 的 字符 串 ， 有 具体 代码 如 下 。 


上 


<?xml version="1.0" encodine="utf-8"?> 
<resources> 
<string name="hello">example19</string> 
<string name="app_name">example19</string> 
<string name="str button1"> 向 左旋 转 </string> 
j 右 旋转 </string> 


<string name="str_done"> 完 成 </string> 


I 


<string name="str_button2"> 


I 


</resources> 


第 四 步 : 编写 处 理 文件 xuanzhuan.java， 其 具体 实现 流程 如 下 。 
1) 加 载 默认 的 Drawable。 

2) 实现 向 左旋 转 按钮 处 理 mButton1.setOnClickListener 事件 。 
3) 实现 向 右 旋转 按钮 处 理 mButton2.setOnClickListener 事件 。 
文件 xuanzhuan.java 的 具体 实现 代码 如 下 : 


package irdc.xuanzhuan; 


| 


忆 


import irdc.xuanzhuan.R; 

import android.app.Activity; 

import android.graphics.Bitmap; 

import android.graphics.BitmapFactory; 
import android.graphics.Matrix; 

import android.graphics.drawable.BitmapDrawable; 
import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 

import android.widget.ImageView; 
import android.widget.TextView; 


public class xuanzhuan extends Activity 
{ 
private Button mButton1; 
private Button mButton2; 
private TextView mTextView!; 
private ImageView mImageView!]; 
private int ScaleTimes; 
private int ScaleAngle; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
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super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButtonl =(Button) findViewById(R.id.myButton!); 

mButton2 =(Button) fmndViewById(R.id.myButton2); 

ImTextViewl = (TextView) fndViewById(R.id.myTextViewl); 
mlmageViewl = (ImageView) fndViewById(R.id.myImageViewl); 
ScaleTimes= 1; 

ScaleAngle = 1; 


final Bitmap mySourceBmp = 
BitmapFactory.decodeResource(getResources(), R.drawable.hippo); 


final int widthOrig = mySourceBmp.getWidth(); 
final int heightOrig = mySourceBmp.getHeight(); 


入 程序 刚 运 行 ， 加 载 默 认 的 Drawable */ 
mlmageViewl.setImageBitmap(mySourceBmp); 


雍 向 左旋 转 按钮 * 
mButtonl.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
public void onClick(View v) 
{ 
ScaleAngle--; 
if(ScaleAngle<—5) 
{ 
ScaleAngle = —5; 
} 


/* ScaleTimes=1， 维 持 1:1 的 宽 高 比例 所 
intnewWidth = widthOrig* ScaleTimes; 
int newHeight = heightOrig * ScaleTimes; 


float scaleWidth = ((float) newWidth) / widthOrig; 
float scaleHeight = ((float) newHeight) / heightOrig; 


Matrix matrix = new Matrix(); 
/* 使 用 Matrix.postScale 设置 维度 */ 
matrix.postScale(scaleWidth, scaleHeight); 


/* 使 用 Matrix.postRotate 方法 旋转 Bitmap*/ 
//matrix.postRotate(S*ScaleAngle); 
matrix.setRotate(S*ScaleAnegle); 
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/* 创建 新 的 Bitmap 对 象 */ 
Bitmap resizedBitmap = 


Bitmap.createBitmap 
(mySourceBmp, 0, 0, widthOrig, heightOrig, matrix, true); 


ah 
BitmapDrawable myNewBitmapDrawable = 
new BitmapDrawable(resizedBitmap); 


mImageViewl.setImageDrawable(myNewBitmapDrawable); 
mTextViewl.setText(Integer.toString(S*ScaleAngle)); 
} 
)); 


全 问 右 旋转 按钮 */ 
mButton2.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
ScaleAngle++; 
if(ScaleAngle>5) 
{ 
ScaleAngle = 5; 


/#ScaleTimes=1， 维 持 1:1 的 宽 高 比例 */ 
int newWidth = widthOrig * ScaleTimes; 
int newHeight = heightOrig * ScaleTimes; 


从 计算 旋转 的 Matrix 比例 */ 
float scaleWidth = ((float) newWidth) / widthOrig; 
float scaleHeight = ((float) newHeight) / heightOrig; 


Matrix matrix = new Matrix(); 
/* 使 用 Matrix.postScale 设置 维度 */ 
matrix.postScale(scaleWidth, scaleHeight); 


/* 使 用 Matrix.postRotate 方法 旋转 Bitmap*/ 
//matrix.postRotate(S*ScaleAngle); 
matrix.setRotate(S*ScaleAnegle); 


/* 创建 新 的 Bitmap 对 象 */ 
Bitmap resizedBitmap = 


Bitmap.createBitmap 
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(mySourceBmp, 0, 0, widthOrig, heightOrig, matrix, true); 
类 


BitmapDrawable myNewBitmapDrawable = 
new BitmapDrawable(resizedBitmap); 


mlmageView!1.setImageDrawable(myNewBitmapDrawable); 
mTextViewl.setText(Integer.toString(S*ScaleAngle)); 


执行 后 将 显示 一 幅 图 片 和 两 个 按钮 ， 分 别 单 击 “左旋 转 ” 和 “ 右 旋转 ”按钮 后 ， 会 实现 
片 旋转 处 理 ， 分 别 如 图 4-36 和 图 4-37 所 示 。 


现 


对 


态 天 僚 7:05A 
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图 4-36 左旋 转 效 果 图 4-37 右 旋转 后 的 效果 


4.]9 decodeFile]| DODODDODODOD 


在 Android 中 ， 如 果 将 图 片 保 存在 “resvdrawable” 目 录 下 ， 则 可 以 通过 R.drawable.id 来 
获取 图 片 的 id， 即 可 以 任意 调用 图 片 素材 。 但 如 果 没 有 将 图 片 素材 保存 在 这 个 目录 下 ， 例 如 
要 想 从 存储 卡 中 获取 一 幅 图 片 时 ， 就 需要 Android API 中 的 Bitmap 对 和 象 来 实现 。 

在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 来 讲解 使 用 decodeFile 方法 来 加 载 手 机 
磁盘 中 图 片 文件 的 方法 。 本 实例 保存 在 “光盘 :\daimaM\example20” 中 ， 其 具体 实现 流程 如 下 。 


Tm. 


a 


第 一 步 : 打开 eclipse， 依 次 单 击 “File” 一 “New” 一 “Android Project”， 新 建 一 个 名 为 
“example20” 的 工程 文件 。 
第 二 步 : 编写 布局 文件 main.xml 实现 整体 布局 ， 分 别 插入 1 个 TextView 组 件 ，1 个 
ImageView 组 件 ，1 个 Button 组 件 。 

第 三 步 : 编写 处 理 文件 example20.java， 使 用 方法 BitmapFactory.decodeFile(fileName) 获 
取 Bitmap 对 象 ， 然 后 用 setImageBitmap() 方 法 设置 将 获取 的 信息 作为 在 ImageView 组 件 中 显 
示 的 图 像 。 文 件 xuanzhuan.java 的 具体 实现 代码 如 下 。 
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package irdc.example20; 


入 加 载 相关 类 */ 
import irdc.example20.R; 


import java.io.File; 

import android.app.Activity; 

import android.graphics.Bitmap; 

import android.graphics.BitmapFactory; 
import android.os.Bundle; 

import android.view.View; 

import android.widget.Button; 

import android.widget.ImageView; 


import android.widget.TextView; 


public class example20 extends Activity 


{ 
/# 声 明 对 象 变量 


private ImageView mImageView; 


private Button mButton; 
private TextView mTextView; 
private String fileName="/data/data/irdc.ex04 22/ex04 22 2.png"; 


/** Called when the activity is first created. */ 
(@Override 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/# 载 入 布局 文件 main.xml */ 


setContentView(R.layout.main); 


入 取得 Button 对 象 ， 并 添加 onClickListener */ 
mButton = (Button)findViewById(R.id.mButton); 
mButton.setOnClickListener(new Button.OnClickListener() 
{ 
public void onClick(View v) 
{ 
旋 取得 对 象 * 
mlmageView = (ImageView)findViewById(R.id.mImageView); 
ImTextView=(TextView)fndViewById(R.id.mTextView); 
族 检查 文件 是 否 存 在 */ 
File f=new File(fileName); 
if(f.exists()) 


{ 
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产生 Bitmap 对 象 ， 并 放 入 mImageView 对 象 中 */ 
Bitmap bm = BitmapFactory.decodeFile(fileName); 
ImImageView.setImageBitmap(bmy); 
ITextView.sSetText(fileName); 

Dy 

else 

{ 
mTextView.setText(" 文 件 不 存在 "); 

} 


4 


执行 后 将 显示 加 载 的 图 片 ， 如 图 4-38 所 示 。 


res/drawable/practice20_1.png 


下 


图 4-38 ”执行 效果 
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作为 一 个 移动 手机 设备 ， 当 然 需要 具备 交互 通信 的 功能 。 手 机 中 的 交互 通信 方式 有 多 
种 ， 例 如 常见 的 通话 、 短 信 、 邮 件 和 蓝牙 等 。 在 本 章 的 内 容 中 ， 将 通过 具体 的 实例 来 详细 讲 
解 Android 中 交互 式 通信 应 用 的 具体 实现 流程 。 希 望 读 者 能 够 举一反三 ， 为 学 习 本 书后 面 知 
识 打 下 基础 。 


5.1 TexMVMew[ 0000 


本 实例 的 源 代 码 保 存在 “光盘 :\daima\5\examplel1”， 下 面 开 始 讲解 本 实例 的 具体 实现 


5.1.1 功能 介绍 
本 实例 的 功能 是 提供 一 个 文本 输入 框 ， 当 用 户 输入 电话 号 码 后 ， 可 以 进行 拨号 处 理 ; 当 
输入 一 个 网 址 后 ， 可 以 登录 到 这 个 地 址 ， 当 输入 邮箱 后 ， 可 以 发 送 邮件 。 
上 述 功能 是 通过 Linkify 对 象 实现 的 ，Linkify 可 以 让 系统 动态 获取 内 容 ， 并 作出 一 个 
判断 ， 判 断 是 电话 、 邮 箱 还 是 网 址 ， 并 随 之 作出 对 应 的 处 理 。 通 过 TextView 和 EditText 交 
互 ， 就 可 以 在 此 显示 自己 输入 的 数据 值 。 在 具体 实现 上 ， 只 需 重 写 EditText.setOnkey 
ListenerO 即 可 实现 。 


5.1.2 具体 实现 
编写 主 程序 文件 examplel.java， 具 体 代码 如 下 。 


package irdc.examplel; 


import irdc.examplel.R; 

import android.app.Activity; 
import android.os.Bundle; 
import android.text.util.Linkify; 
import android.view.KeyEvent; 
import android.view.View; 
import android.widget.EditText; 
import android.widget. TextView; 


public class examplel extends Activity 


{ 
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private TextView mTIextView01; 
private EditText mEditTextO1; 


/** Called when the activity is first created. */ 
(QOverride 

public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mTlextView01 = (TextView)findViewByld(R.id.myTextView!); 
mbEditText01 = (EditText)find ViewById(R.id.myEditText1); 


mEditText01.setOnKeyListener(new EditText.OnKeyListener() 
{ 
(QOverride 
public boolean onKey(View arg0, int argl, KeyEvent arg2) 
{ 
mlextView01.setText(mEditText01.getText()); 
必 判 断 输 入 是 何 种 类 型 ， 并 与 系统 连接 闷 
Linkify.addLinks(mTextView01,Linkify.WEB_ URLSILinkify. 
EMAIL ADDRESSESILinkifyPHONE NUMBERS); 
return false; 


} 


在 上 述 代 码 中 ， 设 置 了 Linkify 的 处 理事 件 ，Linkify 是 在 创建 TextView 之 后 设置 的 ， 否 
则 设置 的 RE 不 会 起 效果 。 并 且 设 置 了 mTextView01 对 和 象 拥有 自动 链接 功能 ， 这 样 就 能 转 到 
指定 的 页 面 。 

执行 后 的 效果 如 图 5-1 所 示 ， 用 户 可 以 输入 电话 号 码 、 邮 箱 地 址 或 网 址 ， 输 入 后 可 显示 
对 应 的 操作 。 例 如 ， 输 入 www.163.com 后 ， 会 在 下 面 自动 显示 输入 的 网 址 ， 如 图 5-2 所 
示 ; 当 单 击 网 址 后 会 来 到 对 应 的 页 面 ， 如 图 5-3 所 示 。 


动 剖 全 1:38 pw 动 剖 全 1:39w 


可 以 输入 电话 、 E-mail 、 网 址 
Www.163.com 


网 址 


WWw.163.com 


图 5-1 执行 效果 图 5-2 自动 显示 输入 数据 
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帐号 密码 选择 去 向 “~ | | 登录 
手机 号 玛 也 可 以 登录 免费 乡 定 

网易 新 闻 军事 评论 图 片 财经 股票 基金 

NETEASE 育 BA 中超 奥运 视频 资讯 综艺 

娱乐 ”电影 电视 音乐 女人 时尚 美容 


www-.163.com 


北京 请 [网 页 | 图 片 ”视频 ”购物 
23C~29C 有 道 Youdao | 
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新 东方 号 恨 班 ， 提 升 一 夏 

2013 公 务 员 考 试 培 训 招生 

北大 汇丰 七 千 私 厂 人 脉 

2012 高 考 招生 志愿 填报 

新 东方 纳什 大 学 北美 直通 ”楼 市 守望 者 如 何 自 处 ” 恒 大 15 周 年 大 庆 
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本 实例 的 源 代码 保存 在 “光盘 :\daima\5\example2”， 下 面 开 始 讲 解 本 实例 的 具体 实现 


5.2.1 功能 介绍 


本 实例 实现 了 一 个 基本 的 手机 拨打 电话 的 过 程 ， 首 先 使 用 了 一 个 EditText 用 于 获取 输入 
的 电话 号 码 ， 当 单 击 Button 后 执行 拨打 电话 程序 ， 并 且 通 过 自 定 义 的 isPhoneNumberValid 
CString phoneNumber) 来 确保 用 户 输入 的 是 合法 的 电话 号 码 。 

在 具体 拨打 电话 时 ， 首 先 要 在 文件 AndiroidManifestxml 中 添加 uses-permission 权限 ， 
这 样 就 实现 了 对 拨打 电话 的 声明 ， 然 后 通过 自 定 义 的 Intent 对 象 ， 通 过 “ACTION_CALL” 
键 和 Uri.parse(0) 方 法 将 用 户 输 入 的 电话 号 码 写 入 ;最 后 ， 通 过 startActivity() 方 法 即 可 完成 程 
序 拨打 电话 的 功能 。 


5.2.2 具体 实现 


本 实例 的 主 程序 文件 是 example2.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 引用 类 和 对 象 ， 具 体 代码 如 下 。 


package irdc.example2; 


import android.app.Activity; 

import android.content. Intent; 

/# 引 用 Uri 类 才能 使 用 Uri.parseO*/ 
import android.net.Uri; 


import android.os.Bundle; 
import android.view.View; 
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人 # 引 用 widgetButton 才能 声明 使 用 Button 对 象 */ 
import android.widget.Button; 


import android.widget. Toast; 
/# 引 用 widget.EditText 才能 声明 使 用 EditText 对 象 */ 
import android.widget.EditText; 
/# 引 用 java.util.regex 才能 使 用 Regular 表达 式 */ 
import irdc.example2.R; 


import java.util.regex.Matcher; 


import java.util.regex.Pattern; 


public class example2 extends Activity 
{ 
此 声明 Button 与 EditText 对 象 名 称 */ 
private Button mButton1; 
private EditText mEditText1; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 通 过 findViewByld 构造 器 来 构造 EditText 与 Button 对 象 */ 
mEditIextl = (EditTexb findViewById(R.id.myEditTextl); 
mButtonl = (Button) fndViewById(R.id.myButton1l); 


iT 


2) 设置 Button 按钮 的 OnClickListener 来 监听 OnClick 事 伯 


由 体 代 码 如 下 。 


mButtonl.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
try 
{ 
放 取 得 EditText 中 用 户 输入 的 字符 串 */ 
String strInput = mEditTextl.getText().toString(); 
if (isPhoneNumberValid(strInput)==true) 
{ 
人 # 建 构 一 个 新 的 Intent 
运行 action.CALL 的 常数 与 通过 Uri 将 字符 申 带 入 */ 
Intent myIntentDial = new 


Intent 


( 


"android.intent.action.CALL", 
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} 


Uri.parse("tel:"+strInput) 

); 

/* 在 startActivity0 方 法 中 

带 入 自 定 义 的 Intent 对 象 以 运行 拨打 电话 的 工作 六 
startActivity(myIntentDial); 
mEditTextl.setText(""); 

} 


else 
{ 
mEditTextl.setText(""); 
Toast.makeText( 
example2.this, "输入 的 电话 格式 不 符 "， 
Toast.LENGTH LONG).show(); 
} 
} 
catch(Exception e) 
{ 


e.printStackTrace(); 


} 


由 


3) 定义 isPhoneNumberValid(String phoneNumben) 方 法 ， 检 查 字 符 串 是 否 为 电话 号 人 码 的 


方法 ， 
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并 返回 true 或 者 false 的 判断 值 。 具 体 代码 如 下 。 


public static boolean isPhoneNumberValid(String phoneNumber) 


{ 


boolean isValid = false; 

从 可 接受 的 电话 格式 有 : 
* AN : 可 以 使 用 "(" 作为 开头 
* (\df{3)): 紧 接 着 三 个 数字 
*W)? : 可 以 使 用 ")" 接 续 
* [- ]?: 在 上 述 格式 后 可 以 使 用 上 选择 性 的 "-". 

* (Wd{4}) : 再 紧 接着 四 个 数字 

*[-]?: 可 以 使 用 具 选 择 性 的 "-" 接续 . 

* (Wd{4})$: 以 四 个 数字 结束 

* 可 以 比较 下 列 数字 格式 : 

* (123)456-78900, 123-4560-7890, 12345678900, (123)-4560-7890 

/ 

String expression = "ANOCd{3DN2L ?Cdf3L ?Cdf1S 3 

String expression2 ="^\(2(N\d {3921- ?OQd{4D[- ?Od{4})8"; 

CharSequence inputStr = phoneNumber; 

/# 创 建 Pattern 对 象 */ 

Pattern pattern = Pattern.compile(expression); 

/* 将 Pattern 以 参数 传 入 Matcher 作 Regular 表达 式 */ 

Matcher matcher = pattern.matcher(inputStr); 


名 
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/# 创 建 Pattern2 对 象 */ 
Pattern pattern2 =Pattern.compile(expression2); 
/* 将 Pattern2 以 参数 传 入 Matcher2 作 Regular expression*/ 
Matcher matcher2= pattern2.matcher(inputStr); 
if(matcher.matches()llmatcher2.matches()) 
{ 
isValid = true; 
} 
return isValid; 
} 
} 


程序 执行 后 的 效果 如 图 5-4 所 示 ; 如 果 输 入 的 电话 号 码 不 规范 则 输出 对 应 的 提示 ， 如 
图 5-5 所 示 ; 当 输 入 规范 的 号 码 并 单 击 “ 开 始 拨打 ”按钮 后 ， 会 实现 拨号 处 理 ， 显 示 拨 打 界 
面 ， 如 图 5-6 所 示 。 


EN 


夫 


团 鲁 4:21rv 
0 输入 的 电话 格式 不 符 
开始 拨打 
图 5-4 执行 效果 图 5-5 ”自动 显示 输入 数据 


1-347-595-9369 


时 Call in progress 


Press Menu for call options 


图 5-6 拨打 界面 
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除了 拨打 电话 外 ， 发 送 短信 也 是 另外 一 个 极为 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 
个 具体 实例 的 实现 ， 介 绍 Android 系统 实现 短信 发 送 功能 的 方法 。 本 实例 的 源 代码 保存 在 
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“光盘 :daimaNSvexample3”， 下 面 开 始 讲解 本 实例 的 具体 实现 流 各 


5.3.1 功能 介绍 
在 本 实例 中 ， 定 义 了 两 个 EditText 


了 判断 手机 号 码 规范 化 的 方法 ， 并 且 设 置 了 短信 的 字数 不 
在 具体 实现 上 ， 是 通过 SmsManage 〈 短 


0 


mm 
O 


控件 ， 用 于 分 别 获 取 收 信人 电话 和 短信 正文 ， 并 设置 


馈 过 70 个 字符 。 


音 管 理 ) 对 象 的 sendTextMessage() 方 法 来 完成 


的 。 在 sendTextMessage0 方 法 中 要 传 入 5 个 值 ， 分 别 是 : 
String、 正 文 String、 发 送 服务 PendingIntent 和 送 达 服务 服务 


5.3.2 具体 实现 


本 实例 的 主 程序 文件 是 example3.java， 下 面 开 始 讲解 其 


收 件 人 地 址 String、 发 送 地 址 
PendingIntent。 


具体 实现 代码 。 


1) 声明 一 个 Button 变量 和 两 个 EditText 变量 ，EditText 供 获取 输入 收 信人 电话 号 码 和 


短信 内 容 ，Button 按钮 用 于 激活 发 信和 处 到 


public class example3 extends Activity 


{ 


E 程 序 。 具 体 代 码 如 1 


这 


/声明 一 个 Button 变量 与 两 个 EditText 变量 */ 


private Button mButton1; 
private EditText mEditTextl; 
private EditText mEditText2; 


/** Called when the activity is first created. */ 


(QOverride 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


名 
* 通过 findViewByld 构造 器 来 


* EditText1，EditText2 与 Button 对 象 


昌 


mbEditTextl=(EditText) fndViewById(R.id.myEditText1l); 
mbEditText2=(EditText) fndViewById(R.id.myEditText2); 
mButton1=(Button) fndViewById(R.id.myButton 1l); 


人 # 将 默认 文字 加 载 EditText 中 */ 
mEditTextl.setText(" 请 输入 号 人 码 "); 
mEditText2.setText(" 请 输入 内 容 !!"); 


/# 设 置 onClickListener 方法 ， 月 


明证 当月 


有 


o 


户 单 击 EditText 时 做 出 反应 */ 


mbEditTextl.setOnClickListener(new EditText.OnClickListener() 


{ 
public void onClick(View v) 


{ 
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/* 单 击 EditText 时 清空 正文 */ 
mEditTextl .setText(""); 
} 
} 
); 


2) 设置 onClickListener() 方 法 ， 用 于 当 用 户 单 击 EditText 时 做 出 反应 。 具 体 代码 如 下 。 


/* 设 置 onClickListener 方法 ， 用 于 当 用 户 单 击 EditText 时 做 出 反应 */ 
mbEditText2.setOnClickListener(new EditText.OnClickListener() 
{ 
public void onClick(View v) 
{ 
/* 单 击 EditText 时 清空 正文 */ 
mbEditText2.setText("™"); 
} 
} 
); 


: 


3) 设置 onClickListener 方法 ， 用 于 用 户 单 击 Button 时 做 出 反应 ， 有 具体 代码 如 下 。 


放 设 置 onClickListener 方法 ， 当 用 户 单 击 Button 时 做 出 反应 六 
mButton1.setOnClickListener(new Button.OnClickListener() 
{ 

(@Override 

public void onClick(View v) 

{ 

庶 由 EditTextl 取得 短信 收 件 人 电话 */ 

String strDestAddress =mEditTextl .getText().toString(); 
入 由 EditText2 取得 短信 文字 内 容 */ 
String strMessage = mEditText2.getText().toString(); 
/# 建 构 一 取得 default instance 的 SmsManager 对 象 */ 


SmsManager smsManager = SmsManager.getDefault(); 


k 


先 检查 收 件 人 电话 格式 与 短信 字数 是 否 超过 70 字符 ， 然 后 通过 smsManagerSendText 
Message 实现 短信 发 送 ， 具 体 实现 代码 如 下 。 

/# 检 查收 件 人 电话 格式 与 短信 字数 是 否 超过 70 字符 */ 
if(isPhoneNumberValid(strDestAddress)==true && 


iswithin70(strMessage)=—true) 


try 
{ 
诺 
* 两 个 条 件 都 检查 通过 的 情况 下 ， 发 送 短信 
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米 


米 


/ 


先 建 构 一 PendingIntent 对 象 并 使 用 getBroadcast0 方 法 广播 


将 PendingIntent， 电 话 ， 短 信 文 字 等 参数 
* 传 入 sendTextMessage0 方 法 发 送 短信 


PendingIntent mPI= PendingIntent.getBroadcast 


(example3.this, 0, new Intent(), 0); 


smsManagetr.sendTextMessage 
(strDestAddress, null, strMessage, mPI, nul]); 


} 


catch(Exception e) 


{ 


e.printStack Trace(); 


} 


Toast.makeText 


( 


example3.this," 送 出 成 功 !1"， 

Toast.LENGTH SHORT 
).show(); 
mbEditTextl.setText(""); 
mbEditText2.setText(""); 


电话 格式 与 短信 文字 不 符合 条 件 时 ， 以 Toast 提醒 光 


} 
else 
{ 
7 
if isPhoneNumberValid(strDestAddress)==false) 
{# 且 字数 超过 70 字符 */ 
if(iswithin70(strMessage)==false) 
{ 
‘Toast.makeText 
( 


} 


example3 .this, 


"电话 号 码 格式 错误 + 短信 内 容 超 过 70 字 ， 请 检查 !1"， 


Toast.LENGTH SHORT 
).Show(); 


else 


{ 
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Toast.makeText 


( 


example3 .this, 


"电话 号 码 格式 错误 ， 请 检查 !1"， 
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Toast.LENGTH SHORT 
).show(); 
} 
} 
/# 字 数 超过 70 字符 */ 
else if (iswithin70(strMessage)==false) 


{ 
Toast.makeText 
( 
example3 .this, 
"短信 内 容 超过 70 字 ， 请 市 除 部 分 内 容 !1"， 
Toast.LENGTH SHORT 
).show); 
} 
} 
} 
)); 


放 检 查 字 符 串 是 否 为 电话 号 码 可 接受 的 格式 ， 并 返回 true 或 false 的 判断 值 */ 
public static boolean isPhoneNumberValid(String phoneNumber) 
{ 
boolean isValid = false; 
从 可 接受 的 电话 格式 有 : 
* AN : 可 以 使 用 "( 作为 开头 
* (\d{3)): 紧 接着 三 个 数字 
* WW)? : 可 以 使 用 ")" 接 续 
* [- ]?: 在 上 述 格式 后 可 以 使 用 上 其 选 择 性 的 "-" 
* (\d{3)) : 再 紧 接着 三 个 数字 
* [- ]?: 可 以 使 用 有 具 选择 性 的 "-" 接续 
* (Wd{5})$: 以 五 个 数字 结束 
* 可 以 比较 下 列 数字 格式 : 
* (123) 456-7890, 123-456-7890, 1234567890, (123)-456-7890 
sh 
String expression = 


NOCdBDN2L RQOABDE OA Ds; 


/* 可 接受 的 电话 格式 有 : 
* 人 ^\(? : 可 以 使 用 "(" 作为 开头 
* (\df3)): 紧 接 着 三 个 数字 
2 
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* [- ]?: 在 上 述 格式 后 可 以 使 用 具 选 择 性 的 "-" 
* (Wd{4}) : 再 紧 接着 四 个 数字 
* [- ]?: 可 以 使 用 具 选 择 性 的 "-" 接续 
* (Wd{4})$: 以 四 个 数字 结束 
* 可 以 比较 下 列 数字 格式 : 
* (02)3456-7890, 02-3456-7890, 0234567890, (02)-3456-7890 
*/ 


String expression2= 


"VENIBDYE ROAADE Cd4D3 


CharSequence inputStr = phoneNumber; 


/创建 Pattern*/ 


Pattern pattern = Pattern.compile(expression); 
/* 将 Pattern 以 参数 传 入 Matcher 作 Regular expression*/ 
Matcher matcher = pattern.matcher(inputStr); 


/创建 Pattern2*/ 


Pattern pattern2 =Pattern.compile(expression2); 
/* 将 Pattern2 以 参数 传 入 Matcher2 作 Regular expression*/ 
Matcher matcher2= pattern2.matcher(inputStr); 
if(matcher.matches()llmatcher2.matches()) 
{ 
isValid = true; 
} 
return isValid; 


public static boolean iswithin70(String text) 
{ 

if (text.length()<= 70) 

{ 


return true; 


} 


else 


{ 


return false; 


» 


在 上 述 代 码 中 ， 通 过 PendingIntent.getBroadcast() 方 法 自 定 义 了 PendingIntent 类 ， 此 类 能 
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够 处 理 即 将 发 生 的 事情 。 然 后 通过 Broadcast 进行 广播 处 理 ， 然 后 使 用 SmsManager 对 象 中 

SmsManager.getDefault() 方 法 来 构建 预 处 理 的 数据 ， 使 用 sendTextMessage() 方 法 将 有 关 的 数 
据 以 参数 形式 带 入 ， 这 样 即 可 完成 发 短信 的 任务 。 

执行 后 的 效果 如 图 5-7 所 示 ， 输 入 手机 号 码 ， 编 写 短信 内 容 后 ， 单 击 “ 发 送 ” 按 钮 后 即 

可 完成 短信 发 送 功能 ， 系 统 会 提示 成 功 信息 。 如 图 5-8 所 示 。 


[CE 


入 5554:aa 


| 局 疯 人 @ zm 


example3 


| 六 交 @ 5m 


请 输入 内 容 !! 


发 送 


图 5-7 初始 效果 图 5-8 发 送 成 功 


如 果 短 信 内 容 和 收 信人 号 码 格式 不 规范 ， 会 输出 对 应 的 错误 提示 。 


5.4 DUOODOESmailDD 


除了 拨打 电话 和 发 送 短信 外 ， 发 送 邮件 也 是 另外 一 个 极为 重要 的 功能 。 在 本 节 的 内 容 
中 ， 将 通过 一 个 具体 实例 的 实现 过 程 ， 介 绍 Android 系统 实现 邮件 发 送 功能 的 方法 。 本 实例 
的 源 代码 保存 在 “光盘 :\daima\5\example4”， 下 面 开 始 讲 解 本 实例 的 具体 实现 流程 。 


5.4.1 ”功能 介绍 
在 本 实例 中 ， 自 定义 了 Intent 类 ， 并 使 用 Android.content.Intent.ACTION_SEND 的 参数 
设置 通过 手机 寄 发 Email 的 服务 ， 整 个 过 程 还 算 简单 。 
在 具体 实现 上 ， 邮 件 的 收发 过 程 是 通过 Android 内 置 的 Gmail 程序 实现 的 ， 并 不 是 使 用 
SMTP 的 Protocol 实现 的 。 为 了 确保 邮件 能 够 发 出 ， 必 须 在 收 件 人 字段 上 输入 标准 的 邮件 地 
址 格式 ， 如 果 格 式 不 规范 ， 则 发 送 按钮 处 于 不 可 用 状态 。 


5.4.2 具体 实现 

本 实例 的 主 程序 文件 是 example4.java， 下 面 开 始 讲解 其 具体 实现 代码 。 

1) 分 别 声明 四 个 EditText、 一 个 Button 以 及 四 个 String 变量 ， 用 于 输入 邮箱 地 址 、 邮 
件 、 主 题 和 副本 ， 有 具体 代码 如 下 。 
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public class example4 extends Activity 

{ 
放声 明 四 个 EditText、 一 个 Button 以 及 四 个 String 变量 */ 
private EditText mEditTextO1; 
private EditText mEditText02; 
private EditText mEditText03; 
private EditText mEditText04; 
private Button mButton01; 


private String[] strEmailReciver; 
private String strEmailSubject; 
private String[] strEmailCc; 
private String strEmailBody ; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
/# 通 过 findViewByld 构造 器 来 建构 Button 对 象 */ 
mButton01 = (Button)findViewById(R.id.myButton!); 
/# 通 过 findViewByld 构造 器 来 构造 所 有 EditText 对 象 */ 
mButton01.setEnabled(false); 
/* 设 置 OnKeyListener， 当 key 事件 发 生 时 进行 反应 */ 
mbEditText01 = (EditText)find ViewById(R.id.myEditText!); 
mbEditText02 = (EditText)find ViewById(R.id.myEditText2); 
mEditText03 = (EditText)findViewById(R.id.myEditText3); 
mbEditText04 = (EditText)find ViewById(R.id.myEditText4); 


2) 定义 setOnKeyListener 方法 ， 如 果 用 户 键入 为 正规 Email 文字 ， 则 按钮 可 用 ， 反 之 则 
按钮 不 可 用 ， 其 体 代 人 码 如 下 。 
必 若 用 户 键入 为 正规 Email 文字 ， 则 设置 按钮 enable〈 可 用 ) 反之 则 disable */ 
ImEditIext01.setOnKeyListenernew EditText.OnKeyListener() 
{ 
(QOverride 
public boolean onKey(View v, int keyCode, KeyEvent event) 


{ 
诬 如 果 是 邮件 地 址 格式 ， 则 按钮 可 按 下 
if(isEmail(mEditText01.getText().toString())) 
{ 
mButton01.setEnabled(true); 
} 
else 
{ 
mButton01.setEnabled(false); 


} 
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return false; 


) 
)); 
3) 定义 onClickListener 响应 按钮 单 击 事件 ， 当 单 击 按钮 后 实现 邮件 发 送 处 理 。 具 体 代 
人 码 如 下 。 


/* 定 义 onClickListener 响应 按钮 */ 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
Intent mEmailIntent = new Intent(android.content. Intent. ACTION SEND); 
mEmailIntent.setType("plain/text"); 


strEmailReciver = new String[|{mEditText01.getText().toString()}; 
strEmailCc = new String[|{mEditText02.getText().toString()}; 
strEmailSubject = mEditText03.getText().toString(); 

strEmailBody = mEditText04.getText().toString(); 


mEmailIntent.putExtra(android.content.Intent.EXTRA EMAIL, strEmailReciver); 
mEmailIntent.putExtra(android.content.Intent.EXTRA CC, strEmailCc); 
mEmailIntent.putExtra(android.content.Intent. EXTRA SUBJECT, strEmailSubject); 
mEmailIntent.putExtra(android.content.Intent.EXTRA TEXT, strEmailBody); 
startActivity(Intent.createChooser(mEmailIntent, getResources().getString(R.string.str message))); 


} 


4) 定义 isEmail(String strEmail) 方 法 ， 检 查 是 否 是 规范 的 邮件 地 址 格式 。 有 其 体 代码 如 下 。 
public static boolean isEmail(String strEmail) 


{ 
String strPattern = "^[a-zA-Z][NwN.-]#[a-zA-Z0-9]@[a-zA-Z0-9][NwN.-]*[a-zA-Z0-9]N[a-zA-Z][a- 
ZzA-Z\.]*[a-zA-Z]$"; 
Pattern p = Pattern.compile(strPattern); 


Matcher m = p.matcher(strEmail); 
return m.matches(); 


} 
} 


执行 后 的 效果 如 图 5-9 所 示 ， 输 入 手机 和 号码， 编写 短信 内 容 后 ， 单 和 
可 完成 短信 发 送 功能 ， 系 统 会 提示 成 功 信 息 。 如 图 5-10 所 示 。 


注意 : Android 模拟 器 中 不 会 内 置 Gmail Client 端 程序 ， 所 以 当 使 用 本 实例 发 送 邮件 
后 ， 会 显示 “No Application can perform this action” 的 提示 。 但 是 在 现实 手机 设备 上 ， 如 果 


“发 送 ”按钮 后 即 


1 


Lt 


面 图 155 
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运行 本 实例 程序 ， 会 调用 Gmail 程序 ， 成 功 实 现 邮 件 发 送 。 


国人 @ 2:23PM 


人 正在 发 信 中 .. 


No applications can perform 
this action. 


图 5-9 ”初始 效果 图 $-10 发送 成 功 
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除了 基本 的 通话 和 短信 外 ， 震 动 也 是 另外 一 个 极为 重要 的 功能 。 通 过 手机 震动 ， 能 够 帮 
助 我 们 及 时 感知 打 来 的 电话 或 发 来 的 短信 ， 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 过 
程 ， 介 绍 Android 系统 实现 手机 震动 功能 的 方法 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\5\ 


example5”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


5.5.1 实现 原理 

在 本 实例 中 ， 读 者 将 会 了 解 到 触发 手机 震动 事件 的 方法 。Android 中 的 震动 事件 是 
Vibration， 需 要 设置 震动 的 时 间 长 短 和 周期 ， 设 置 单位 是 毫秒 。 如 果 要 建立 手机 震动 ， 则 必 
须 建 立 Vibrator 对 象 ， 并 通过 调用 Vibrate() 方 法 来 实现 震动 目的 。 在 Vibrator 构造 器 中 有 4 
个 参数 ， 前 三 个 用 于 设置 震动 大 小 ， 最 后 那个 用 于 设置 震动 持续 时 间 。 

在 本 范例 中 的 震动 方式 是 不 同 的 ， 分 为 一 直 持 续 和 只 震动 一 轮 两 种 。 


5.5.2 具体 实现 

本 实例 的 主 程序 文件 是 example5.java， 下 面 开始 讲解 其 具体 实现 代码 。 

1) 设置 ToggleButton (开关 ) 的 对 象 ， 检 测 ToggleButton 是 否 被 启动 ， 如 果 单 让 
“ON ”按钮 则 启动 震动 模式 ， 如 果 单 击 “OFF” 按 钮 则 关闭 震动 模式 。 具 体 代码 如 下 。 


上 以 


public class example5 extends Activity 


{ 


private Vibrator mVibrator01; 
/** Called when the activity is first created. */ 
(QOverride 


public void onCreate(Bundle savedInstanceState) 
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super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


人 # 设 置 ToggleButton 对 象 */ 
mVibrator01 = ( Vibrator )getApplication().getSystemService 
(Service.VIBRATOR SERVICE); 


final ToggleButton mtogglebutton1 = 
(ToggleButton) findViewById(R.id.myTogglebutton!1); 


final ToggleButton mtogglebutton2 = 
(ToggleButton) findViewById(R.id.myTogglebutton2); 


final ToggleButton mtogglebutton3 = 
(ToggleButton) findViewById(R.id.myTogglebutton3); 


2) 设置 短 震动 模式 ， 通 过 mVibrator01.vibrate( new long[]{100,10,100,1000},-]) 设 置 了 震 
动 周期 ， 具 体 代码 如 下 。 


I 
mtogglebutton1.setOnClickListener(new OnClickListener() 
{ 
public void onClick(View v) 
{ 
if (mtogglebutton1.isChecked()) 
{ 
/# 设置 震动 的 周期 */ 
mVibrator01.vibrate( new long[]{100,10,100,1000},-1); 
信用 Toast 显示 震动 启动 */ 
‘Toast.make Text 


( 


example5.this, 


getString(R.string.str_ok), 
Toast.LENGTH SHORT 
).show(); 
} 
else 
{ 
入 取消 震动 */ 
mVibrator01.cancel(); 
人/# 用 Toast 显示 震动 已 被 取消 */ 
Toast.makeText 


( 


example5.this, 


getString(R.string.str_end), 
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Toast.LENGTH SHORT 
).Show(); 
} 
} 


人 


3) 设置 长 震动 模式 ， 通 过 mvVibrator01.vibrate(new long[]{100,100,100,1000},0); 设 置 了 震 
动 周 期 ， 具 体 代 码 如 下 。 


启发 殿 动 5 
mtogglebutton2.setOnClickListener(new OnClickListener() 
{ 
public void onClick(View v) 
{ 
if (mtogglebutton2.isChecked()) 
{ 
证 设置 震动 的 周期 */ 
mVibrator01.vibrate(new long[|{100,100,100,1000},0); 


人 # 用 Toast 显示 震动 启动 */ 
‘Toast.makeText 


( 


example5.this, 


getString(R.string.str_ok), 
Toast.LENGTH SHORT 
).show(); 
} 
else 
{ 
入 取消 震动 */ 
mVibrator01.cancel(); 


/#* 用 Toast 显示 震动 取消 */ 
‘Toast.makeText 
( 
example5.this, 
getString(R.string.str_ end), 
Toast.LENGTH SHORT 
).Show(); 
} 
) 
); 


4) 设置 节奏 震动 模式 ， 通 过 mVibrator01.vibrate( new long[]{1000,50,1000,50,1000},0); 设 
置 了 震动 周期 ， 具 体 代 码 如 下 。 


入 节奏 震动 */ 
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mtogglebutton3.setOnClickListener(new OnClickListener() 


{ 
public void onClick(View v) 
{ 
if (mtogglebutton3.isChecked()) 
{ 
人 # 设置 震动 的 周期 */ 


mVibrator01.vibrate( new long[]{1000,50,1000,50,1000},0); 


/* 用 Toast 显示 震动 启动 */ 
ToastmakeText 
( 
example5 .this, getString(R.string.str_ok), 
Toast.LENGTH SHORT 
).show(); 
} 
else 
{ 
和 # 取消 震动 % 
mVibrator01.cancel(); 
/* 用 Toast 显示 震动 取消 */ 
‘Toast.makeText 
( 


example5 .this, 


getString(R.string.str_end)， 
Toast.LENGTH SHORT 
).show(); 


另外 ， 还 需要 在 文件 AndroidManifest.xml 中 设置 权限 ， 
VIBRATE 的 权限 ， 有 具体 代码 如 下 。 


<manifest 


xmlns:android="http://schemas.android.com/apk/res/android" 


android:versionCode="1" 
android:version Name="1.0.0" package='"irdc.example5"> 
<application 
android:icon="(@drawable/icon" 
android:label="(@string/app_name"> 
<activity 
android:label="(@string/app_name" 
android:name=" 
<intent-filter> 


.example5"> 


即 必须 允许 Android.permission. 
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<action android:name="android.intent.action.MAIN" /> 


<category android:name="android.intent.category.LAUNCHER" /> 
</intent-filter> 
</activity> 


</application> 


<uses-permission android:name="'"android.permission.VIBRATE" /> 


</manifest> 


执行 后 的 效果 如 图 5-11 所 示 ， 当 选择 一 种 模式 ， 并 单 击 按钮 后 会 局 动 对 应 的 震动 模 


式 ， 如 图 5-12 所 示 的 是 启动 了 “ 短 时 间 ” 震 动 模式 。 


选择 振动 模式 


26 DDOD 


图 5-11 


在 本 书 前 面 的 内 容 
醒目 的 提示 效果 ， 提 示 / 


P， 己 经 i 


初始 效果 


解 过 Toast 提醒 
户 即将 需要 完成 的 操作 。 在 本 节 的 内 容 


对 


3°512 


短 时 间 震 动 模型 


的 基本 知识 。Toast 能 够 在 手机 中 实现 一 个 


Ph， 将 通过 一 个 具体 实例 的 


实现 ， 介 绍 Android 系统 实现 图 文 提 醒 功 能 的 方法 。 本 实例 的 源 代 码 保存 在 “光盘 :daiman 


Sexample6”， 下 面 3 


5.6.1 实现 原理 


于 始 


解 本 实例 的 


只 体 实 现 流程 。 


功能 。 图 文 提 醒 和 单独 的 文字 提醒 相 比 ， 具 有 更 好 的 视觉 效果 。 


文字 ， 这 些 图 片 和 文字 就 是 提醒 的 内 容 。 
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在 本 实例 中 ， 在 Toast 中 放置 一 幅 图 片 ， 并 且 辅 助 了 一 行文 字 ， 这 样 即 可 实现 图 文 提醒 


在 具体 实现 上 ， 在 Toast 中 放置 一 个 Layout〔 布 局 界面 )， 这 个 Layout 中 包含 了 图 片 和 
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5.6.2 具体 实现 
本 实例 的 主 程序 文件 是 example6java， 下 面 开 始 讲解 其 具体 实现 代码 。 文 件 
example6.java 的 具体 实现 流程 如 下 。 
1) 设置 Button 用 OnClickListener 启动 事件 setOnClickListener(new Button.OnClickListener()。 
2) 分 别 创建 ImageView 对 象 和 LinearLayout 〈 线 性 布局 ) 对 象 。 
3) 设置 mTextView 去 抓 取 string 值 。 
4) 判断 mTextView 的 内 容 为 何 ， 并 与 系统 实现 连接 。 
5) 用 Toast 方式 将 内 容 显示 出 来 。 
文件 example6.java 的 具体 代码 如 下 。 


T 


上 


public class example6 extends Activity 


{ 


private Button mButton01; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
mButton01 = (Button)findViewById(R.id.myButton!); 


人 # 设 置 Button 用 OnClickListener 启动 事件 */ 
mButton01.setOnClickListener(new Button.OnClickListener() 


{ 


(QOverride 
public void onClick(View v) 


{ 
/TODO Auto-generated method stub 


/# 创建 ImageView 对 象 */ 
ImageView mView01 =mnew ImageView(example6.this); 


TextView mTextView = new TextView(example6.this); 


/* 创建 LinearLayout 对 象 */ 


LinearLayout lay = new LinearLayout(example6.this); 


/* 设置 mTextView 去 抓 取 string 值 */ 
mTextView.setText(R.string.app_utl); 


加 图 161 


Android 开发 完全 实战 宝典 


执行 后 的 效果 如 
醒 ， 如 图 5-14 所 示 。 


162 看 看 


全 判断 mTextView 的 内 容 为 何 ， 并 与 系统 做 连接 */ 
Linkify.addLinks 
( 
mTlextView,Linkify. WEB URLS| 
Linkify.EMAIL ADDRESSES| 
Linkify.PHONE NUMBERS 
); 


作用 Toast 方式 显示 信息 */ 
Toast toast = Toast.makeText 
( 
example6.this, 
mTextView.getText(), 
ToastLENGTH LONG 


); 


/* 自 定义 View 对 象 */ 


View textView = toast.getView(); 


访 以 水 平方 式 排列 *%/ 
lay.setOrientation(LinearLayout.HORIZONTALD); 


/* 在 ImageView Widget 里 指定 显示 的 图 片 *%/ 


mView01.setImageResource(R.drawable.icon); 


/# 在 Layout 里 添加 刚 创建 的 View */ 
layaddView(mView01); 


/* 在 Toast 里 显示 提示 文字 */ 
lay.addView(textView); 


/* 以 Toast，setView 方法 将 界面 传 入 */ 
toast.setView(lay); 


/* 显示 Toast */ 


toast.show(); 


而 


Ek 
mm 
Ln 

| 


5S—13 所 示 ， = 


击 “ 有 图 片 的 提醒 ”按钮 后 ， 会 弹出 


四 
个 


的 提 
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图 5-13 初始 效果 图 5-14 ”图 文 提醒 
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在 手机 的 最 顶端 ， 常 用 于 显示 时 间 、 信 号 强度 和 电池 用 量 ， 这 就 是 状态 栏 。 在 实现 提 
醒 处 理 时 ， 也 可 以 在 状态 栏 中 显示 一 幅 图 片 ， 并 结合 图 片 和 文字 来 实现 更 加 美观 的 提示 。 
在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 状态 栏 中 实现 图 文 提醒 功 
能 的 方法 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\5\example7”， 下 面 开始 讲解 本 实例 的 具体 
实现 流程 。 


5.7.1 ”实现 原理 


在 本 实例 中 ， 在 状态 栏 中 放置 一 幅 提醒 图 标 。 在 具体 实现 上 ，Android API 为 了 管理 提 
示 信 息 ， 定 义 了 NotificationManage (通知 管理 对 象 )， 只 需 将 Notification 〈 通 知 信 息 ) 添加 
到 NotificationManage 即 可 将 信息 显示 在 状态 栏 。 


5.7.2 具体 实现 


本 实例 的 主 程序 文件 是 example7.java 和 example7 1.java， 下 面 先 讲解 文件 
example7.java 的 具体 实现 代码 。 

1) 先 声 明 对 象 变 量 ， 分 别 设置 在 线 、 离 开 、 忙 碌 、 一 会 回来 和 离线 5 种 状态 。 具 体 代 
码 如 下 。 


public class example7 extends Activity 


{ 
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放声 明 对 象 变 量 */ 
private Notification Manager myNotiManager; 
private Spinner mySpinner; 
private ArrayA dapter<String> myA dapter; 
private static final String[] status = 
和 "在线 "离开 "忙碌 "一 会 回来 "离线 " }; 


(@Override 
protected void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/# 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


2) 初始 化 NotificationManager 对 象 ， 然 后 使 用 myspinner_dropdown 自 定 义 下 拉 菜 单 
模式 ， 并 将 ArrayAdapter 添加 Spinner 对 和 象 中 ， 根 据 具体 状态 显示 对 应 的 提示 图 标 。 具 体 
代码 如 下 。 


雍 初始 化 对 象 */ 
myNotiManager= 
(Notification Manager)getSystemService( NOTIFICATION SERVICE); 
mySpinner=(Spinner)findViewById(R.id.mySpinner); 
myAdapter=new ArrayAdapter<String>(this, 


android.R.layout.simple spinner item,status); 
上/# 应 用 myspinner dropdown 自 定义 下 拉 菜 单 模式 */ 
myAdapter.setDropDownViewResource(R.layout.myspinner dropdown); 
上 将 ArrayAdapter 添加 Spinner 对 象 中 */ 
mySpinner.setAdapter(myAdapter); 


/* 将 mySpinner 添加 OnItemSelectedListener */ 
mySpinner.setOnItemSelectedListener( 
new Spinner.OnltemSelectedListener() 


{ 
(QOverride 


public void onItemSelected(AdapterView<?> arg0,View argl, 
int arg2,long arg3) 


此 依照 选择 的 item 来 判断 要 发 哪 一 个 提醒 */ 
if(status[arg2].equals(" 在 线 ")) 
{ 
setNotiType(R.drawable.msn," 在 线 "); 
} 
else if(status[arg2].equals(" 离 和 开 ")) 
{ 
setNotiType(R.drawable.away," 亢 和 开 "); 
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else ifstatus[arg2].equals(" 忙 碌 中 由 ) 


{ 
setNotiType(R.drawable.busy" 忙 碌 中 ; 
} 
else if(status[arg2].equals(" 马 上 回来 ")) 
{ 
setNotiType(R.drawable.min," 蕊 上 回来 "); 
} 
else 
{ 
setNotiType(R.drawable.offine," 离 线 "); 
} 


} 


3) 定义 onNothingSelected(AdapterView<?> arg0)， 供 用 户 选择 状态 。 具 体 代 码 如 下 。 


(QOverride 
public void onNothingSelected(AdapterView<?> arg0) 


)); 


/* 发 出 Notification 的 方法 */ 
private void setNotiType(int iconId, String text) 
{ 
/* 创建 新 的 Intent， 作 为 单 击 Notification 留言 条 时 ， 
* 会 运行 的 Activity */ 
Intent notifyIntent=new Intent(this,example7 1.class); 
notifyIntent.setFlags( Intent.FLAG ACTIVITY NEW_TASK); 
匀 创建 PendingIntent 作为 设置 递 延 运行 的 Activity */ 
PendingIntent appIntent=PendingIntent.getActivity(example7 .this,0,notifyIntent,0); 


4) 创建 Notication (通知 〉 对 象 ， 并 设置 相关 对 应 的 参数 。 具 体 代 人 码 如 下 。 


Notification myNoti=new Notification(); 


/* 设置 statusbar 显示 的 icon */ 


myNoti.icon=iconld; 
/* 设置 statusbar 显示 的 文字 信息 */ 
myNoti.tickerText=text; 

/* 设置 notification 发 生 时 间 时 发 出 默认 声音 */ 
myNoti.defaults=Notification.DEFAULT_SOUND,; 

旋 设置 提醒 留言 条 的 参数 */ 
myNoti.setLatestEventInfo(example7.this,"MSN 登录 状态 " 


TGS ， 


text,appIntent); 
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/* 送出 Notification */ 
myNotiManager.notify(0,myNoti); 


} 
} 


然后 介绍 文件 example7_1.java， 其 功能 是 引入 example7_1 extends Activity， 并 发 出 
Toast 提示 ， 实 现 模 拟 QQ 和 MSN 的 登录 效果 。 具 体 代码 如 下 。 


package irdc.example7; 


/import 相关 类 * 
import android.app.Activity; 
import android.os.Bundle; 


import android.widget. Toast; 


入 当 用 户 单 击 提醒 留言 条 时 ， 会 运行 的 Activity */ 
public class example7 1 extends Activity 


{ 
(@Override 


protected void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


/* 发 出 Toast */ 
‘Toast.makeText(example7 1 .this, 
"模拟 MSN 切换 登录 状态 "， 
ToastLENGTH SHORT 
).Show(); 
finish(); 


} 


执行 后 将 首先 显示 默认 的 “在 线 ” 状 态 ， 并 在 状态 栏 中 显示 QQ 的 在 线 图 标 ， 如 图 5-15 
所 示 。 可 以 在 下 拉 框 中 继续 选择 一 种 状态 ， 对 应 的 状态 栏 也 会 显示 对 应 的 图 标 ， 如 图 5-16 
所 示 。 例 如 ， 当 选择 “离开 ”按钮 后 ， 会 在 状态 栏 中 显示 对 应 的 提示 图 标 ， 如 图 5-17 所 
示 。 这 是 一 个 MSN 的 图 标 。 


Si 名 而 全 257m 在 线 
离开 
忙碌 
现在 的 状态 : 
和 一 会 回来 
- .离线 
图 5-15 初始 效果 图 5-16 图 文 提醒 
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号 离线 


和 
example7 


现在 的 状态 : 
忙碌 


图 5-17 离开 图 标 


5.8 ContentResolverl| D00000D 


在 手机 应 用 中 ， 联 系 人 检索 是 一 个 十 分 常见 的 功能 之 一 ， 用 户 可 以 通过 手机 的 查询 系 
统 ， 迅 速 找到 通讯 录 中 的 联系 人 电话 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 
通过 ContentResolver 实现 检索 手机 通讯 录 的 方法 。 本 实例 的 源 代 码 保存 在 “ 光 
盘 :\daimaNSvexample8”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


5.8.1 实现 原理 
在 百度 主页 中 ， 只 需 输 入 要 检索 关键 字 的 首 字符 ， 系 统 就 能 够 提示 显示 出 很 多 对 应 的 关 


键 字 。 同 样 ， 在 手机 联系 人 检索 应 用 中 ， 也 可 以 实现 类 似 的 功能 效果 。 即 输入 要 查询 的 联系 
人 信息 ， 系 统 能 够 根据 输入 的 关键 字 首 字符 ， 上 自动 显示 出 通讯 录 中 对 应 的 联系 人 信息 。 


5.8.2 ”ContentResolver 介 绍 


Android 是 如 何 实现 应 用 程序 之 间 数 据 共享 的 ”一 个 应 用 程序 可 以 将 自己 的 数据 完全 暴 

露出 去 ， 外 界 根本 看 不 到 ， 也 不 用 看 到 这 个 应 用 程序 暴露 的 数据 是 如 何 存储 的 ， 或 者 是 使 用 
数据 库 还 是 使 用 文件 ， 还 是 通过 网 上 获得 ， 这 些 一 切 都 不 重要 ， 重 要 的 是 外 界 可 以 通过 这 
套 标准 及 统一 的 接口 和 这 个 程序 里 的 数据 打交道 ， 例 如 ， 添 加 (insert)、 删 除 〈delete )、 查 
询 (query) 或 修改 (update )， 当 然 需 要 一 定 的 权限 才 可 以 。 
如 何 将 应 用 程序 的 数据 暴露 出 去 ? Android 提供 了 Content Provider， 一 个 程序 可 以 通 
过 实现 一 个 Content Provider 的 抽象 接口 将 自己 的 数据 完全 暴露 出 去 ， 而 且 Content 
providers 是 以 类 似 数据 库 中 表 的 方式 将 数据 暴露 。Content Providers 存储 和 检索 数据 ， 通 
过 它 可 以 让 所 有 的 应 用 程序 访问 数据 ， 这 也 是 应 用 程序 之 间 唯 一 共享 数据 的 方法 。 要 想 使 
应 用 程序 的 数据 公开 化 ， 可 通过 两 种 方法 : 创建 一 个 属于 你 自己 的 Content Provider 或 者 将 
你 的 数据 添加 到 一 个 已 经 存在 的 Content Provider 中 。 前 提 是 我 的 数据 有 相同 数据 类 型 并 且 
有 写 入 Content Provider 的 权限 。 

如 何 通 过 一 套 标 准 及 统一 的 接口 获取 其 他 应 用 程序 暴露 的 数据 ? Android 提供 了 
ContentResolver， 外 界 的 程序 可 以 通过 ContentResolver 接口 访问 Content Provider 提供 的 
数据 。 
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在 下 面 的 内 容 中 ， 将 着 重 讲解 如 何 获取 其 他 应 用 程序 共享 的 数据 ， 比 如 获取 Android 手 
机 电话 籍 中 的 信息 。 
1. 什么 是 URI 
URI 是 网 络 资源 的 定义 ， 在 Android 中 赋予 其 更 广阔 的 含义 ， 先 看 个 例子 ， 如 下 ， 


content://com.example.transportationprovider/trains/122 
一 一 一 一 
A B CcC DD 


在 上 例 中 ， 将 其 分 为 A，B，C, D 共 4 个 部 分 。 

A: 标准 前 级 ， 用 来 说 明 一 个 Content Provider 控制 这 些 数据 ， 无 法 改变 的 。 

B: URI 的 标识 ， 它 定义 了 是 哪个 Content Provider 提供 这 些 数据 。 对 于 第 三 方 应 用 程 
序 ， 为 了 保证 URI 标识 的 唯一 性 ， 它 必须 是 一 个 完整 的 、 小 写 的 类 名 。 这 个 标识 在 
<provider> 元 素 的 authorities 属性 中 说 明 。 


<provider name=”.TransportationProvider” 


上 


authorities= ”com.example.transportationprovider > 


C: 路 径 ，Content Provider 使 用 这 些 路 径 来 确定 当前 需要 什么 类 型 的 数据 ，URI 中 可 能 
不 包括 路 径 ， 也 可 能 包括 多 个 ; 

D: 如 果 URI 中 包含 ， 表 示 需 要 获取 的 记录 的 ID; 如 果 没 有 ID， 就 表示 返回 全 部 ; 
由 于 URI 通常 比较 长 ， 而 且 有 时 候 容 易 出 错 ， 且 难以 理解 。 所 以 ， 在 Android 当中 定义 
了 一 些 辅助 类 ， 并 且 定 义 了 一 些 常 量 来 代替 这 些 长 字符 串 ， 例 如 ，People.CONTENT _URI。 

2. ContentResolver 简 介 

ContentResolver 是 通过 URI 来 查询 Content Provider 中 提供 的 数据 。 除 了 URI 外 ， 还 必 
须知 道 需要 获取 的 数据 段 的 名 称 ， 以 及 此 数据 段 的 数据 类 型 。 如 果 你 需要 获取 一 个 特定 的 记 
录 ， 就 必须 知道 当前 记录 的 IDP， 也 就 是 URI 中 了 部 分 。 
前 面 也 提 到 了 Content Providers 是 以 类 似 数据 库 中 表 的 方式 将 数据 暴露 出 去 ， 那 么 
ContentResolver 也 将 采用 类 似 数据 库 的 操作 来 从 Content Providers 中 获取 数据 。 现 在 简要 介 
绍 ContentResolver 的 主要 接口 ， 具 体 如 表 5-1 所 示 。 


表 5-1 ContentResolver 返回 值 


返 回信 函数 声明 
final Uri insert(Uri url, ContentValues values)Inserts a row into a table at the given URL. 
final int delete(Uri url, String where, String[] selectionArgs)Deletes row(s) specified by a content URI. 


query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder)Query the given URL 


final Cursor : 
Cc returning a Cursor over the result set. 


final int update(Uri uri, ContentValues values, String where, String[] selectionArgs)Update row(s) in a content URI. 


3. ContentResolver 简 单 实例 
通过 上 面 内 容 的 学 习 ， 了 解 了 如 何 获取 、 使 用 ContentResolver 的 方法 。 下 面 启动 
Eclipse， 制 作 一 个 ContentResolver 简单 实例 。 

1) 打开 Eclipse， 创建 一 个 名 为 “ContentResolver” 的 项 目 ， 如 图 5-18 所 示 。 

2) 编写 主 程序 代码 ， 有 具体 如 下 。 
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Project rame: |ContentProvider 


Contents 
© Create new project in workspace 
Octreate Project from existine source 


MUse default location 


Build Target 


Target Hame Vendor Flatform AF... 


DMmdroid 1.1 Android Dpen Source Project 1 二 2 
回 Amdroid 1.5 Android Dpen Source Project I 3 
加 GooglLe AFIs Google Inc. ea 时 


Android + Google AEIs 


Properties 


Application name: moandroid. showcontact 


Package name; moandroid. showcontact 


[Vcreate Activity: |showcontact 


Min SDK Version: 3 


图 5-18 新建 项 目 


package moandroid.showcontact; 
import android.app.ListActivity; 
import android.database.Cursor; 
import android.os.Bundle; 


import android.provider.Contacts.Phones; 

import android.widget.ListAdapter; 

import android.widget.SimpleCursorAdapter; 

public class showcontact extends ListActivity { 
protected void onCreate(Bundle savedInstanceState) { 


super.onCreate(savedInstanceState); 

Cursor c = getContentResolver().query(Phones.CONTENT _URL null, null, null, null); 
startManagingCursor(c); 

ListAdapter adapter = new SimpleCursorAdapter(this, 
android.R.layout.simple list item 2, c， 

new String[] { Phones.NAME, Phones.NUMBER }， 

new int[] { android.R.id.text1, android.R.id.text2 }); 

setListAdapter(adapter); 

} 

} 


3) 在 AndroidManifest.XML 中 <application> 元 素 前 增加 如 下 许可 ， 代 码 如 下 。 


<uses-permission android:name=”android.permission.READ CONTACTS” /> 


最 后 运行 程序 ， 在 模拟 器 启动 后 ， 单 击 Menu 返回 到 Home 界面 ， 打开 Contacts 选择 
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Contacts 标签 页 ， 添 加 两 个 联系 人 信息 。 返 
刚 添加 的 两 个 联系 人 信息 将 显示 在 界面 上 。 


5.8.3 ”具体 实现 

经 过 前 面 知 识 的 学 习 ， 读 者 对 ContentResolver 的 知识 又 有 了 更 进一步 的 认识 ， 下 面 开 始 
介绍 本 节 范 例 的 具体 实现 过 程 。 本 实例 的 主 程序 文件 是 example8.java 和 Contacts 
Adapterjava。 

1. 主 程序 文件 example8.java 

下 面 先 讲解 文件 example8.java 的 具体 实现 代码 。 

1) 通过 String[] 获 取 通 讯 录 中 的 字段 ， 具 体 代 码 如 下 。 

public class example8 extends Activity 


{ 


private AutoCompleteTextView myAutoComplete TextView; 


3 


到 Home， 选 择 moandroid.showcontact 运行 ， 


private TextView myTextView!; 

private Cursor contactCursor; 

private ContactsAdapter myContactsA dapter; 

入 要 获取 通讯 录 的 字段 */ 

public static final String[] PEOPLE PROJECTION =new String[] 

{ Contacts.People. ID, Contacts.People.PRIMARY _ PHONE ID, 
Contacts.People.TYPE, Contacts.People.NUMBER, Contacts.People.LABEL., 
Contacts.People.NAME }; 


2) 通过 ContentResolver content = getContentResolver(); 和 contactCursor = content.query 来 
获取 通讯 录 里 的 数据 ， 具 体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


myAutoCompleteTextView = (AutoCompleteTextView) 
findViewById(R.id.myAutoCompleteTextView); 
myTextViewl1 = (TextView) fndViewById(R.id.myTextView1l); 


/* 定义 ContentResolver 对 象 */ 
ContentResolver content = getContentResolver(); 


谍 取得 通讯 录 的 焦点 * 
contactCursor = content.query 


( 
Contacts.People.CONTENT_URL， 


PEOPLE PROJECTION, null, null, 
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Contacts.People. DEFAULT SORT ORDER 
); 


诺 将 Cursor 传 入 自己 实现 的 ContactsAdapter 容器 */ 
myContactsAdapter = new ContactsAdapter(this, contactCursor); 


myAutoCompleteTextView.setAdapter(myContactsAdapter); 


在 上 述 代 码 是 以 Cursor 来 获取 通讯 录 中 的 存储 内 容 的 ， 例 如 电话 号 码 和 姓名 等 。 

3) 定义 myAutoCompleteTextView. setOnItemClickListener 事件 ， 即 当 用 户 单 击 联系 人 姓 
名 后 的 拦截 事件 处 理 ， 并 通过 ContactsAdaptergetCursor0 来 获取 联系 人 的 电话 号 码 。 有 具体 代 
码 如 下 。 


ImyAutoCompleteTextView.setOnItemClickListener 
(new AdapterView.OnItemClickListener() 
{ 
(QOverride 
public void onItemClick 
(AdapterView<?> arg0, View argl, int arg2, long arg3) 
{ 
和 # 取得 焦点 沁 
Cursor c = myContactsAdapter.getCursor(); 
谍 移 到 所 单 击 的 位 置 */ 


c.moveToPosition(arg2); 


String number = c.getString 
(c.getColumnIndexOrThrow(Contacts.People.NUMBER)); 
诺 当 找 不 到 电话 时 显示 无 输入 电话 */ 

number = number 二 null ? "无 输入 电话 " : number; 
ImyTextView1l.SetIext(c.getString 
(c.getColumnIndexOrThrow(Contacts.People.NAME)) 

+ "的 电话 是 " +number); 


DE 
} 
2. 主 程序 文件 ContactsAdapter.java 
接 下 来 看 文件 ContactsAdapterjava， 在 此 定义 了 继承 于 CursorAdapt 的 类 Contacts- 
Adapter， 并 以 下 拉 数 据 对 象 curse 作为 下 拉 荣 单数 据 的 类 ， 并 且 重 写 了 runQueryOn 
BackgroundThread(CharSequence constraint)， 设 置 当 输入 * 时 ， 显 示 出 所 有 的 联系 人 电话 信 
恩 。 具 体 代码 如 下 。 


public class ContactsAdapter extends CursorAdapter 


{ 


private ContentResolver mContent; 


public ContactsAdapter(Context context, Cursor c) 
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{ 


super(context, c¢); 


mContent = context.getContentResolver(); 


@Override 
public void bindView(View view, Context context, Cursor cursor) 
{ 
旋 取得 通讯 录 人 员 的 名 字 六 
((TextView) view).setText(cursor.getString(cursor 
.getColumnIndexOrThrow(Contacts.People.NAME))); 


@Override 
public View newView(Context context, Cursor cursor, ViewGroup parent) 
{ 
final LayoutInflater inflater = LayoutInflater.from(context); 
final TextView view = (TextView) inflater.inflate( 
android.R.layout.simple dropdown item lline, parent, false); 
view.setText(cursor.getString(cursor 
.getColumnIndexOrThrow(Contacts.People.NAME))); 


return view; 


(@Override 


public String convertToString(Cursor cursor) 


{ 


return cursor.getString(cursor.getColumnIndexOrThrow(Contacts.People.NAME)); 


(@Override 
public Cursor runQueryOnBackgroundThread(CharSequence constraint) 
{ 

if (getFilterQueryProvider() != null) 

{ 


return getFilterQueryProvider().runQuery(constraint); 


StringBuilder buffer = null; 
String[] args = null; 
if (constraint != null) 


{ 
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buffer = new StringBuilder(); 
buffer.append("UPPER("); 
buffer.append(Contacts.ContactMethods.NAME); 
buffer.append(") GLOB ?"); 
args =new String[] 
{ constraint.toString().toUpperCase() + "*" }; 
} 
return mContent.query(Contacts.People.CONTENT _ URI, 
example8.PEOPLE PROJECTION, buffer == null ? null : buffer.toString(), 
args, Contacts.People.DEFAULT SORT ORDER); 


} 


3. 打开 权限 
在 文件 AndroidManifest.xml 中 ， 需 要 建 打 开 访 问 通讯 录 的 权限 ， 具 体 代 码 如 下 。 


<manifest xmlns:android="http://schemas.android.com/apk/res/android" 


package="irdc.example8" 


android:versionCode="1" 
android:versionName="1.0.0"> 
<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 
<activity android:name=".example8" 
android:label="(@string/app_name"> 
<intent-filter> 
<action android:name="android.intent.action.MAIN" /> 
<category android:name="android.intent.category. LAUNCHER" /> 
</intent-filter> 
</activity> 
</application> 
<uses-permission android:name="android.permission.READ CONTACTS"></uses-permission> 


</manifest> 


执行 后 的 初始 效果 如 图 5-19 所 示 。 当 输入 一 个 联系 人 字符 后 ， 系 统 能 够 根据 联系 人 自 
动 提 示 显 示 对 应 的 信息 ， 如 图 5-20 所 示 。 当 输入 “*” 后 会 显示 通讯 录 中 所 有 联系 人 的 信 
息 ， 如 图 5-21 所 示 。 


碟 天 全 3:01rm 1 
lexamples Eee 
TT 
to shili8 
图 $-19 ”初始 效果 图 $-20 ”提示 的 信息 
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图 5-21 所 有 联系 人 信息 


二 


在 手机 应 用 中 ， 经 常 利 用 手机 来 存储 各 种 各 样 的 文件 信息 ， 这 样 就 需要 一 个 很 好 的 工具 
来 实现 对 各 个 文件 的 管理 。 在 本 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 Android 实 
现 文件 管理 的 基本 流程 。 本 实例 的 源 代 码 保存 在 “光盘 :daimaNS\example9”， 下 面 开 始 讲解 
本 实例 的 具体 实现 流程 。 


5.9.1 ”实现 原理 


在 本 实例 中 ， 通 过 ListActivity (列表 视图 组 件 ) 和 Java LIO 来 查找 根 目 录 下 的 所 有 文 
件 ， 并 通过 setListAdapter( ) 方 法 将 存放 文件 信息 的 ArrayAdapter 对 象 传递 给 ListView 〔〈 列 表 
视图 )。 而 Android API 提供 的 ArrayAdapter 对 象 只 允许 存 入 String 数组 或 List 对 象 ， 为 此 在 
显示 文件 列表 时 只 能 以 字符 串 的 形式 来 显示 文件 名 。 如 果 要 同时 显示 文件 名 、 图 标 和 文件 夹 
名 ， 则 需要 定义 一 个 实现 Adapter 接口 的 对 象 。 
在 Android API 中 提供 了 BaseAdapter 对 象 ， 只 要 继承 了 此 对 象 ， 就 可 以 实现 属于 自己 的 
Adapter， 在 本 实例 中 ， 事 先 准 备 了 素材 图 片 作为 图 标 ， 用 于 代表 不 同 的 文件 或 文件 夹 类 型 。 


5.9.2 具体 实现 


本 实例 的 主 程序 文件 是 example9.java 和 MyAdapterjava。 

1. 主 程序 文件 example8.java 

下 面 先 讲解 文件 example9.java 的 具体 实现 代码 。 

1) 声明 需要 的 变量 ， 其 中 items 表示 存放 显示 的 名 称 ，paths 表示 存放 文件 路 径 ， 
rootPath 表示 起 始 目录 。 有 具体 代码 如 下 。 


public class example9 extends ListActivity 


{ 

此 变量 声明 

items: 存放 显示 的 名 称 

paths: 存放 文件 路 径 

rootPath: 起 始 目录 中 
private List<String> items=null; 
private List<String> paths=null; 
private String rootPath="/"; 
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private TextView mPath; 


2) 载 入 主 布局 文件 main.xml， 然 后 初始 化 mPath 用 于 显示 目前 路 径 。 具 体 代码 如 下 。 


(QOverride 
protected void onCreate(Bundle icicle) 


{ 


super.onCreate(icicle); 


/# 载 入 main.xml Layout */ 
setContentView(R.layout.main); 
入 初始 化 mPath， 用 以 显示 目前 路 径 *% 
mpPath=(TextView)findViewByld(R.id.mPath); 
getFileDir(rootPath); 

} 


3) 定义 方法 getFileDir(String filePath)， 用 于 获取 文件 的 具体 架构 。 具 体 代码 如 下 。 


入 取得 文件 架构 的 方法 光 
private void getFileDir(String filePath) 
{ 


庆 设置 目前 所 在 路 径 */ 
mpPath.setText(filePath); 
items=new ArrayList<String>(); 
paths=new ArrayList<String>(); 
File 会 new File(filePath); 

File[ | files=f.listFiles(); 


if(!filePath.equals(rootPath)) 

{ 
旋 第 一 笔 设 置 为 [ 回 到 根 目录 ] */ 
items.add("b1"); 
paths.add(rootPath); 
入 第 二 笔 设置 为 [ 回 到 上 一 层 ] */ 
items.add("b2"); 
paths.add(f.getParent()); 

} 

谨 将 所 有 文件 添加 ArrayList 中 *%/ 

for(int 二 0;i<files.length;i++) 

{ 
File file=files[i]; 
items.add(file.getName()); 
paths.add(file.getPath()); 


} 


匀 使 用 自 定义 的 MyAdapter 来 将 数据 传 入 ListActivity */ 
setListAdapter(new MyAdapter(this,items,paths)); 
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4) 定义 按钮 触发 事件 处 理 方法 onListItemClick， 如 果 是 文件 夹 就 再 运行 getFileDir()， 
如 果 是 文件 就 运行 openFile()。 具 体 代 码 如 下 。 
庆 设置 ListItem 被 单 击 时 要 做 的 动作 */ 
@Override 


protected void onListItemClick(ListView 1,View v,int position,long id) 


{ 
File file=new File(paths.get(position)); 
if (file.isDirectory()) 
{ 
访 如 果 是 文件 夹 就 再 运行 getFileDir() */ 
getFileDir(paths.get(position)); 


} 

else 

{ 
雍 如 果 是 文件 就 运行 openFileO */ 
openFile(file); 

} 


} 
5) 定义 方法 openFile(File 有， 用 于 在 手机 上 打开 指定 的 文件 。 具 体 代码 如 下 。 


ER 

private void openFile(File f) 

{ 
Intent intent = new Intent(); 
intent.addFlags(Intent.FLAG ACTIVITY NEW _ TASK); 
intent.setAction(android.content.Intent.ACTION VIEW); 


Lh 


人 # 调用 getMIMEType0 来 取得 MimeType */ 
String type = get MIMEType(D); 

上 六 设置 intent 的 file 与 MimeType */ 
intent.setDataAndType(Uri.fromFile(f),type); 
startActivity(intent); 


} 


6) 定义 getMIMEType(File 全 方法 ， 用 于 判断 文件 的 类 型 。 其 中 “end” 是 结尾 的 扩展 
名 。 只 需 获取 “end” 即 可 。 具 体 代码 如 下 。 


/# 判断 文件 MimeType 的 方法 */ 
private String getMIMEType(File f) 
{ 

String type 

String fName=f.getName(); 

上 # 取得 扩展 名 */ 

String end=fName.substring(fName.lastIndexOf(".")+1, 

fName.length()).toLowerCase(); 


2 
和” 
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具 依附 档 名 的 类 型 决定 MimeType */ 
if(end.equals("m4a")llend.equals("mp3")llend.equals("mid") 


{ 


} 


type = 


llend.equals("xmf'')llend.equals("ogg")llend.equals("wav")) 


A 


audio"; 


else if(end.equals("3gp")llend.equals("mp4")) 


{ 


} 


type = 


= 


Video"; 


else if(end.equals("jpg")llend.equals("gif"')llend.equals("png") 


{ 


} 


else 


{ 


} 


llend.equals("jpeg")llend.equals("bmp")) 


type = "image"; 


从 如 果 无 法 直接 打开 ， 就 跳出 软件 列表 给 用 户 选择 */ 
type="*"; 


type 十 一 二 
return type; 


} 
} 


2. 文件 MyAdapterjava 
接 下 来 讲解 文件 MyAdapterjava 的 具体 实现 代码 。 


1) 分 别 声明 如 下 4 个 变量 。 


口 mIconl : 
口 mIcon2 : 
口 mIcon3 : 
口 mIcon4: 


回 到 根 目录 的 图 文件 。 
回 到 上 一 层 的 图 档 。 

文件 夹 的 图 文件 
文件 的 图 档 。 


o 


package irdc.example9; 


旋 加 载 相关 类 */ 

import irdc.example9.R; 

/# 自 定义 的 Adapter， 继 承 android.widget.BaseAdapter */ 
public class MyAdapter extends BaseAdapter 


{ 


族 变量 声明 * 
private LayoutInflater mInflater; 


private Bitmap mlcon!l; 


private Bitmap mlcon2; 
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private Bitmap mlcon3; 
private Bitmap mlcon4; 
private List<String> items; 
private List<String> paths; 


2) 定义 MyAdapter 构造 器 ， 并 分 别传 入 了 items、paths 和 mInflater 这 3 个 参数 ， 最 后 
对 前 面 定义 的 4 个 变量 进行 了 赋值 。 上 有 具体 代码 如 下 。 
/* MyAdapter 的 构造 器 ， 传 入 3 个 参数 *%/ 
public MyAdapter(Context context,List<String> it,List<String> pa) 
{ 
入 参数 初始 化 忆 
mInfater=LayoutInflater.from(context); 


items = it; 

paths = pa; 

ImIconl = BitmapFactory.decodeResource(context.getResources(),R.drawable.back01); 
mlcon2 = BitmapFactory.decodeResource(context.getResources(),R.drawable.back02); 
ImIcon3 = BitmapFactory.decodeResource(context.getResources(),R.drawable.folder); 
ImIcon4 = BitmapFactory.decodeResource(context.getResources(),R.drawable.doc); 


} 


3) 分 别 覆 六 3 个 方法 : getCount()、getltem(int position) 和 getItemlId(int position)。 具 体 
代码 如 下 。 


/#* 因 继 承 BaseAdapter， 需 覆盖 以 下 方法 */ 
(QOverride 
public int getCount() 


{ 


return items.size(); 


(QOverride 
public Object getItem(int position) 
{ 


return items.get(position); 


} 


(QOverride 
public long getItemId(int position) 


{ 
return position; 


} 
4) 使 用 上 自 定义 的 fle_row 作为 布局 ， 然 后 分 别 设置 “ 回 到 根 目录 ”的 文字 与 icon (图 
标 格式 ) 和 “ 回 到 上 一 层 ” 的 文字 与 icon。 具 体 代码 如 下 。 


(QOverride 
public View getView(int position, View convertView,ViewGroup parent) 


人 


1 
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ViewHolder holder; 


if(convertView == null) 
{ 
/* 使 用 自 定义 的 fle row 作为 Layout */ 
convertView = mInflater.inflate(R.layout.file_row, null); 
/* 初始 化 holder 的 text 与 icon */ 
holder = new ViewHolder(); 
holder.text = (TextView) convertView.find ViewBylId(R.id.text); 
holdericon = (ImageView) convertView.findViewById(R.id.icon); 


convertView.setTag(holder); 
} 
else 
{ 
holder = (ViewHolder) convertView.getTag(); 
} 


File 全 new File(paths.get(position).toString()); 
/x* 设置 [ 回 到 根 目录 ] 的 文字 与 icon */ 
if(items.get(position).toString().equals("b1")) 
{ 
holder.text.setText("Back to /"); 
holdericon.setImageBitmap(mIcon1l); 
} 
/* 设置 [ 回 到 上 一 层 ] 的 文字 与 icon */ 
else if(items.get(position).toString().equals("b2")) 
{ 
holder.text.setText("Back to .."); 
holder.icon.setImageBitmap(mIcon2); 
} 
谨 设置 [文件 或 文件 来 ] 的 文字 与 icon */ 
else 
{ 
holder.text.setText(f.getName()); 
if(f.isDirectory()) 
{ 
holder.icon.setImageBitmap(mIcon3); 
} 
else 
{ 
holder.icon.setImageBitmap(mIcon4); 
} 
} 


return convertView; 
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/* class ViewHolder */ 


private class ViewHolder 


{ 
TextView text; 
ImageView icon; 
} 
9 


执行 后 的 初始 效果 如 图 5-22 所 示 。 当 选择 一 个 文件 夹 后 会 继续 显示 此 文件 夹 里 面 的 内 


容 ， 如 图 5-23 所 示 。 


咏 剖 大 3:12rw 


/sdcard 


ea to/ 


Ce) 
已- ea to .. 


520 000000 


图 5-22 ”初始 效果 图 5-23 ”文件 信息 


| 
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PP， 有 一 个 默认 的 开机 主 界面 ， 如 图 5-24 所 示 。 
码 天 全 3:13rm 


图 5-24 默认 主 界面 
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实际 上 ， 可 以 通过 编程 的 方法 来 对 默认 主 界面 进行 控制 。 在 本 节 的 内 容 中 ， 将 通过 一 个 


O 


5.10.1 ”实现 原理 


ContextWrapper 类 的 clearWallpaper0 方 法 ， 这 样 将 


具体 实例 的 实现 ， 介 绍 分 别 清理 和 还 原 主 界面 的 方法 。 本 实例 的 源 代 人 码 保存 在 “光盘 :daima 
\S\example10”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 


在 本 实例 中 ， 首 先 在 文件 te se xml 设置 这 作 权 限 。 办， 后 重 写 了 


5.10.2 具体 实现 


iT 


本 实例 的 主 程序 文人 
1) 设置 Button 按钮 用 OnClickListener 来 启动 事件 ， 具 体 代 人 码 如 下 。 


public class example10 extends Activity 
{ 
private Button mButton1; 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButtonl =(Button) findViewById(R.id.myButton!1); 


此 设置 Button 用 OnClickListener 来 启动 事件 */ 
mButtonl.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
try 
{ 
/* 清 除 背 景 图 案 */ 
clearWallpaper(); 
Toast.makeText(example10.this, getString(R.string.str_ done) 
,Toast.LENGTH SHORT).show(); 


) 
catch (IOException e) 


{ 


e.printStackTrace(); 


} 


2 
) 


是 example10.java， 下 面 开 始 讲解 解 其 具体 实现 流程 


口 


Eo 


2) 


II 


EE 写 clearWallpaper(0) 方 法 ， 清 除 用 户 当 前 的 设置 桌面 ， 从 而 还 原 为 原来 的 默认 设 
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置 。 具 体 代码 如 下 。 
(@Override 


public void clearWallpaper() throws IOException 


{ 
super.clearWallpaper(); 


} 


} 
接 下 来 需要 在 文件 AndroidManifest.xml 中 设置 permission 权限 ， 具 体 代码 如 下 。 


<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
package='"irdc.example10" 
android:versionCode="1" 
android:version Name="1.0.0"> 
<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 
<activity android:name=".example10" 
android:label="(@string/app_name"> 
<intent-filter> 
<action android:name="android.intent.action.MAIN" /> 
<category android:name="android.intent.category.LAUNCHER" /> 
</intent-filter> 
</activity> 
</application> 
<uses-permission android:name="android.permission.SET WALLPAPER"/> 


</manifest> 


执行 后 的 初始 效果 如 图 5-25 所 示 。 当 单 击 “删除 背景 ”按钮 后 ， 会 还 原 到 默认 的 背景 
主 界面 。 


‘examplei10 


example10 


图 5-25 ”初始 效果 


3 村 四 | 四 吧 辐 | 加 


在 手机 应 用 中 ， 可 以 对 手机 的 背景 图 片 进行 置换 处 理 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具 
体 实 例 的 实现 ， 介 绍 Android 背景 图 片 置换 处 理 的 基本 流程 。 本 实例 的 源 代码 保存 在 “ 光 


砚 


pn 
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盘 :\daima\s\example11”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


5.11.1 ”实现 原理 

在 本 实例 中 ， 首 先 在 界面 中 设置 一 个 Gallery 组 件 ， 然 后 利用 Gallery 的 特性 实现 了 图 片 
的 滑动 处 理 ， 最 后 将 Gallery 中 的 Drawable 图 片 设置 为 手机 桌面 。 

在 有 具体 设计 时 ， 需 要 将 置换 的 图 片 必须 和 手机 屏幕 的 大 小 比例 保持 一 致 ， 即 宽 和 高 的 比 
例 一 致 。 


5.11.2 具体 实现 


本 实例 的 主 程序 文件 是 examplell.java， 下 面 开 始 讲解 基 
1) 设置 设置 用 于 置换 的 图 片 素材 名 ， 分 别 如 下 。 
DD drawable\google.png。 


具体 实现 代码 。 


MA 


口 drawable\helloking.png。 
口 drawable\chamberlain.png。 
口 drawable\king.png。 


口 drawable\with.png。 
号 体 代 码 如 下 所 示 。 


public class examplell extends Activity 


{ 


protected static InputStream is; 


private ImageAdapter mImageAdapter01; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


A* 设 置 图 档 */ 

Integer[] myImagelds = 

{ 
R.drawable.google, 
R.drawable.helloking, 
R.drawable.chamberlain, 
R.drawable.king, 
R.drawable.with, 


2) 定义 ImageAdapter 对 象 mImageAdapter01， 然 后 设置 图 的 显示 方式 是 Gallery， 最 后 
分 别 设置 弹出 窗口 的 图 式 、 弹 出 窗口 的 信息 和 确认 窗口 。 具 体 代码 如 下 。 


R 


mImageAdapter01 = new ImageAdapter(examplell1 .this, myImageIds); 
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人 # 设 置 图 为 Gallery 的 显示 方式 */ 
Gallery g=(Gallery) fndViewById(R.id.mygallery); 
g.setAdapter(mImageAdapter01); 


g.setOnItemClickListener(new Gallery.OnItemClickListener() 
{ 
Override 
public void onItemClick 
(AdapterView<?> parent, View v, final int position, long 1d) 
{ 
newAlertDialog.Builder(examplell.this) 
.SetTitle(R.string.app_ about) 
旋 设 置 弹出 窗口 的 图 式 */ 
.SetIcon(mImageAdapter01.myImageIds[position]) 
旋 设 置 弹 出 窗口 的 信息 */ 
.setMessage(R.string.app_about msg) 
庶 确 认 窗 口 */ 
.setPositiveButton(R.string.str_ok, 


new DialogInterface.OnClickListener() 
{ 
public void onClick( 
DialogInterface dialoginterface, int 1) 
{ 
Resources resources = getBaseContext().getResources(); 
js = resources.openRawResource 
(mImageAdapter01.myImageIds[position]); 
try 
{ 


3) 通过 setWallpaper(is) 来 更 换 桌 面 ， 用 Toast 来 显示 桌面 已 更 换 。 
setWallpaper(is); 
/# 用 Toast 来 显示 桌面 已 更 换 */ 
Toast.makeText 


( 


examplel1.this, 


getString(R.string.my gallery text pre), 
Toast.LENGTH SHORT 
).show(); 
} 
catch (Exception e) 
{ 
e.printStack Trace(); 
中 
} 


}).setNegativeButton 
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具体 代码 如 下 。 
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(R.string.str no, new DialogInterface.OnClickListener() 
{ 
4) 设置 跳出 窗 返回 事件 处 理 方法 onClick(DialogInterface dialoginterface, int i)， 然 后 
用 Toast 来 显示 桌面 设置 已 取消 。 有 具体 代码 如 下 。 


户 设 置 跳出 窗口 的 返回 事件 汶 
public void onClick(DialogInterface dialoginterface, int 1) 


{ 


es 


一 < 


人 # 用 Toast 来 显示 桌面 已 取消 */ 
Toast.makeText 


( 


examplel1.this, 


getString(R.string.my_ gallery_text_no), 
Toast.LENGTH SHORT 
).Show(); 
} 
}).showO); 


-> 
} 


5) 定义 方法 ImageAdapter(Context c，Integer[] aid)， 用 于 保存 素材 医 


getCount、getItem 和 getItemld 方法 来 获取 具体 是 哪 幅 图 片 。 具 体 代 码 如 下 。 


唤 
下 
nh 
阅 
注 


public class ImageAdapter extends BaseAdapter 
{ 

int mGalleryItemBackground; 

private Context mContext; 


private Integer[] myImageIds; 


public ImageAdapter(Context c, Integer[] aid) 
{ 
mContext = ¢; 
mylmagelds = aid; 
TypedArray a = obtainStyledAttributes(R.styleable.Gallery); 
mGalleryItemBackground = a.getResourceld 
( 
R.styleable.Gallery android galleryItemBackground, 0 
); 


a.recycle(); 


Override 
public int getCount() 


{ 


return myImageIds.length; 
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》 


Override 
public Object getItem(int positiom) 


{ 


return null; 


Override 
public long getItemId(int positiom) 
{ 


return position; 


} 


[mh 


Ca 


Hz 


6) 分 别 产 生 ImageView 对 象 ， 设 置 图 片 给 imageView 对 象 ， 重 新 设置 界面 的 宽 高 ， 习 
新 设置 图 片 的 宽 高 ， 设 置 Gallery 背景 图 ， 最 后 返回 ImageView 对 象 。 具 体 代 码 如 下 。 


(QOverride 
public View getView 


(int position, View convertView, ViewGroup parent) 


{ 


# 定 义 ImageView 对 象 */ 

ImageView i= new ImageView(mContext); 

放 设 置 图片 给 imageView 对 象 */ 
i.setImageResource(myImagelds[position]); 

必 重 新 设置 图 片 的 宽 高 */ 
i.setScaleType(ImageView.ScaleType.FIT XY); 
/* 重 新 设置 Layout 的 宽 高 交 

i.setLayoutParams(new Gallery.LayoutParams(138, 108)); 
/* 设 置 Gallery 背景 图 */ 
i.setBackgroundResource(mGalleryIltemBackground); 
/# 返 回 ImageView 对 象 */ 


return i; 


(QOverride 
public void setWallpaper(InputStream data) throws IOException 


{ 
super.set Wallpaper(data); 


} 


执行 后 的 初始 效果 如 图 5-26 所 示 。 此 时 可 以 选择 一 幅 
弹出 提示 框 供 用 户 确 认 ， 如 图 5-27 所 示 。 
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图 片 作 为 手机 屏幕 背景 ， 首 先 将 
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图 5-26 初始 效果 图 5-27 文件 信息 


当 单 击 图 5-27 中 的 “确定 ”按钮 后 ， 此 幅 图 片 会 成 为 手机 屏幕 的 背景 ， 如 网 5-28 
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图 $-28 选中 的 图 片 作为 屏幕 背景 
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在 手机 应 用 中 ， 除 了 能 够 查看 文件 和 文件 夹 外 ， 还 需要 能 够 对 文件 进行 一 些 常见 的 操 
作 。 在 本 贡 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 对 手机 内 文件 进行 修改 和 删除 的 基 
本 流程 。 本 实例 的 源 代 码 保存 在 “光盘 :daimaNS\example12”， 下 面 开 始 讲解 本 实例 的 具体 实 
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现 流 程 。 

5.12.1 实现 原理 

在 本 实例 基于 前 面 的 5.9 中 的 实例 ， 即 先 实现 5.9 中 文件 浏览 功能 ， 然 后 再 添加 文件 修 


改 和 删除 功能 ， 实 现 对 指定 文件 或 文件 夹 的 名 字 修 改 或 删除 操作 。 上 述 的 名 字 修 改 和 删除 操 
作 ， 是 通过 Java IO 实现 的 。 


5.12.2 Java I/O 基本 类 库 介 绍 
IO 类 库 通 常 使 用 “ 流 ” 这 个 抽象 概念 ， 它 代表 任何 有 能 力 产 生 数据 的 数据 源 对 象 或 者 
是 有 能 力 接收 数据 的 接收 端 对 象 。“ 流 ”屏蔽 了 实际 的 IO 设备 中 处 理 数 据 的 细节 。 例 如 下 
面 的 代码 ， 
BufferedReader in =new BufferedReader( 
new FileReader("IOStreamDemo.java")); 
BufferedReader stdin = new BufferedReader( 


new InputStreamReader(System.in));// java.io.BufferedReader(java.io.Reader) 
/而 System.in 是 一 个 输入 流 ， 因 此 用 InputStreamReader 进行 转换 (适配器 模式 》 


1) 要 想 打 开 一 个 文件 用 于 字符 输入 ， 可 以 使 用 以 String 或 File 对 象 作 为 文件 名 的 
FileReader。 为 了 提高 速度 ， 和 希望 对 那个 文件 进行 缓冲 ， 那 么 将 作为 结果 的 引用 传 给 一 个 
BufferedReader 构造 器 。 例 如 下 面 的 代码 ， 


BufferedReader in =new BufferedReader(new FileReader("IOStreamDemo.java")); 


String s, s2 = new String(); 

while ((s= in.readLine()) != null) 
Ss2+= s+"\n"; 

in.close(); 


注意 : BufferedReader 也 提供 readLine() 方 法 ， 所 以 这 是 我 们 的 最 终 对 象 和 进行 读 取 的 
接口 。 


另外 ， 也 可 以 用 System.in 读 取 来 自控 制 台 的 输入 。System.in 是 一 个 InputStream， 而 
BufferedReader 需要 的 是 Reader 参数 ， 因 此 引入 InputStreamReader 来 进行 转换 (适配器 模 
式 )。 例 如 下 面 的 代码 ， 


BufferedReader stdin = new BufferedReader(new InputStreamReader( System.in)); 
System.out.println("Enter a line"); 


System.out.printin(stdin.readLine()); 


2) 从 内 存 读 入 。 用 字符 串 创建 一 个 StringReader。 例 如 下 面 的 代码 ， 


// 2. Input from memory 

StringReader in2 = new StringReader(s2); 
int c; 

while((c= in2.read()) != -1) 
System.out.print((char)ce); 
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3) 格式 化 的 内 存 输 入 。 
如 果 要 读 取 格 式 化 数据 (JDK5.0 Documentation: read primitive Java data types)， 此 时 要 
用 到 DataInputStream， 它 是 一 个 面向 字 节 的 IO 类 。 例 如 下 面 的 代码 ， 


DataInputStream in3 = new DataInputStream(new ByteArrayInputStream(s2.getBytes())); 
while(true) 
System.out.print((char)in3.readByte()); 


4) 文件 输出 
首先 创建 一 个 与 指定 文件 连接 的 FileWriter。 通 常会 用 BufferedWriter 将 其 包装 起 来 用 以 
缓冲 输出 。 然 后 为 了 格式 化 把 它 转换 成 PrintWriter。 例 如 下 面 的 代码 ， 


BufferedReader in4 = new BufferedReader(new StringReader(s2)); 

PrintWriter outl = new PrintWriter(new Buffered Writer(new File Writer("D:\IODemo.txt"))); 
int lineCount = 1; 

while(((s = in4.readLine()) != null)) 

outl.printin(lineCountt++ + ": "+ s); 

outl.close(); 


为 了 序列 化 一 个 对 象 ， 首 先 要 创建 菜 些 OutputSteam 对 象 ， 然 后 将 其 封装 在 一 个 
ObjectOutputStream 对 象 内 。 这 时 ， 只 需 调用 writeObjectO 即 可 将 对 象 序列 化 ， 并 将 其 发 送 给 
OutputStream 。 要 将 一 个 序列 重组 为 一 个 对 象 ， 需 要 将 一 个 InputStream 封装 在 
ObjectInputStream 内 ， 然 后 调用 readObject()。 


5.12.3 ”具体 实现 


本 实例 的 主 程序 文件 是 example12.java 和 MyAdapterjava， 下 面 开 始 讲 角 
代码 。 
1. 文件 example12.java 
下 面 先 讲解 文件 example12.java 的 具体 实现 代码 。 
1) 声明 需要 的 三 个 对 象 。 
口 items: 存放 显示 的 名 称 
口 paths: 存放 文件 路 径 
口 rootPath: 起 始 目 录 
人体 代 码 如 下 。 


public class example12 extends ListActivity 
{ 
人 对 象 声 明 
items: 存放 显示 的 名 称 
paths: 存放 文件 路 径 
rootPath: 起 始 目录 
A 
private List<String> items=null; 


其 具体 实现 


- 


private List<String> paths=null; 


加 国 189 


Android 开发 完全 实战 宝典 
private String rootPath="/"; 
private TextView mPath; 
private View myView; 
private EditText myEditText; 


(@Override 
protected void onCreate(Bundle icicle) 
{ 

super.onCreate(icicle); 

/# 载 入 main.xml Layout */ 


setContentView(R.layout.main); 
入 初始 化 mPath， 用 以 显示 目前 路 径 */ 
mpPath=(TextView)findViewByld(R.id.mPath); 
getFileDir(rootPath); 

} 


2) 定义 方法 getFileDir(String filePath)， 用 于 获取 手机 内 的 文件 架构 。 具 体 过 程 如 下 。 
第 一 步 : 设置 目前 所 在 路 径 。 

第 二 步 : 分 别 实现 “ 回 到 根 目 录 ” 和 “ 回 上 层 ” 操 作 。 

第 三 步 : 使 用 自 定 义 的 MyAdapter 来 将 数据 传 入 ListActivity。 

也 体 代 码 如 下 。 


private void getFileDir(String filePath) 
{ 
族 设置 目前 所 在 路 径 * 
mpPath.setText(filePath); 
items=new ArrayList<String>(); 


paths=new ArrayList<String>(); 


File 人 new File(filePath); 
File[] files=f.listFiles(); 


if(!filePath.equals(rootPath)) 

{ 
/# 第 一 笔 设 置 为 [ 回 到 根 目录 ] */ 
items.add("b1"); 
paths.add(rootPath); 
/# 第 二 笔 设置 为 [ 回 上 层 ] */ 
items.add("b2"); 
paths.add(f.getParent()); 

} 

谍 将 所 有 文件 添加 ArrayList 中 党 

for(int i=0;i<files.length;i++) 

{ 
File file=files[i]; 
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items.add(file.getName()); 
paths.add(file.getPath()); 
} 


使 用 自 定义 的 MyAdapter 来 将 数据 传 入 ListActivity */ 
setListAdapter(new MyAdapter(this,items,paths)); 
} 


3) 设置 Listltem( 列 表 项 ) 被 单 击 时 要 做 的 动作 处 理事 件 onListItemClick(ListView 
LView Vint position,long id)， 如 果 是 文件 夹 就 再 运行 getFileDir0 方 法 ， 如 果 是 文件 则 调用 
fileHandle0 方 法 。 具 体 代 码 如 下 。 


(QOverride 


protected void onListItemClick(ListView 1,View v,int position,long id) 
{ 
File file = new File(paths.get(position)); 


if(file.isDirectory()) 

{ 
席 如 果 是 文件 夹 就 再 运行 getFileDir0 方 法 忆 
getFileDir(paths.get(position)); 


} 
else 
{ 
/# 如 果 是 文件 则 调用 fileHandle0 方 法 */ 
fileHandle(file); 
} 
} 
4) 定义 处 理 文件 的 和 eHandle(final File file) 方 法 ， 具 体 过 程 如 下 。 
第 一 步 : 获取 文件 的 OnClickListener 事件 监听 。 


全 一 4 上 上 


第 二 步 : 如 果 which 一 0， 则 选择 要 打开 的 文件 ， 如 果 which==1， 则 修改 文件 名 ; 如果 
which 是 其 他 值 ， 则 删除 选中 文件 。 
中 体 代 码 如 下 。 


庆 处 理 文件 的 方法 */ 
private void fileHandle(final File file){ 
谍 单 击 文件 时 的 OnClickListener */ 
OnClickListener listenerl=new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog,int which) 
{ 
if(which==0) 
{ 
雍 选择 的 item 为 打开 文件 */ 
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openFile(file); 

} 

else if(which==1) 

{ 
雍 选择 的 item 为 更 改 文件 名 */ 
LayoutInflater factory=LayoutInflater.from(example12.this); 
/# 初始 化 myChoiceView， 使 用 rename alert_dialog 为 layout */ 
myView=factory.inflate(R.layout.rename alert dialog,null); 
myEditText=(EditText)myView.findViewById(R.id.mEdit); 
庶 将 原始 文件 名 先 放 入 EditText 中 沁 
myEditText.setText(file.getName()); 


入 新 建 一 个 更 改 文件 名 的 Dialog 的 确定 按钮 的 监听 % 
OnClickListener listener2= 


new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog, int which) 
{ 
入 取得 修改 后 的 文件 路 径 */ 
String modName=myEditText.getText().toString(); 
final String pFile=file.getParentFile().getPath()+"/"; 
final String newPath=pFile+modName; 


上 # 判断 文件 名 是 否 已 存在 */ 
if(new File(newPath).exists()) 
{ 
和 排除 修改 文件 名 时 没 修 改 直 接送 出 的 状况 沁 
if(!ImodName.equals(file.getName())) 
{ 
从 跳出 Alert 警告 档 名 重复 ， 并 确认 是 否 修 改 */ 
new AlertDialog.Builder(example12.this) 
.setTitle(" 注 意 !") 
.SetMessage(" 文 件 名 已 经 存在 ， 是 否 要 和 窗 盖 ?") 
.SetPositiveButton(" 确 定 "， 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog,int which) 
{ 
旋 档 名 重复 仍然 修改 会 覆盖 掉 已 存在 的 文件 光 
file.renameTo(new File(newPath)); 
作 重新 产生 文件 列表 的 ListView */ 
getFileDir(pFile); 
} 
上 


.SetNegativeButton(" 取 消 "， 
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new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog,int which) 
{ 
} 
.show(); 
} 
} 
else 
{ 


放 文件 名 不 存在 ， 直 接 做 修改 动作 */ 
file.renameTo(new File(newPath)); 

入 重新 产生 文件 列表 的 ListView */ 
getFileDir(pFile); 


lh 


} 
本 


/* create 更 改 文件 名 时 跳出 的 Dialog */ 
AlertDialog renameDialog= 

new AlertDialog.Builder(example12.this).create(); 
renameDialog.setView(myView); 


谍 设置 更 改 文件 名 单 击 确认 后 的 Listener */ 
renameDialog.setButton(" 确 定 ",listener2); 
renameDialog.setButton2(" 取 消 "， 
new DialogInterface.OnClickListener() 
{ 

public void onClick(DialogInterface dialog, int which) 


及; 
renameDialog.show(); 
} 
else 
{ 
雍 选择 的 item 为 删除 文件 */ 
new AlertDialog. Builder(example12.this).setTitle(" 注 意 !") 
.setMessage(" 确 定 要 删除 文件 吗 ?") 
.SetPositiveButton(" 人 确定"， 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog, 
int which) 


TT 
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5) 当 ) 
具体 代码 如 下 。 


T 


Ver 
file.delete(); 
getFileDir(file.getParent()); 
} 
D) 
.SetNegativeButton(" 取 消 "， 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog,int which) 
{ 
) 
)).Sshow0); 
} 
} 
间 


} 


String[] menu={" 打 开 文 件 "," 更 改 文件 名 "," 删 除 文 件 "}; 
new AlertDialog.Builder(examplel2.this) 
.setTitle(" 你 要 做 甚么 ?") 
.SetItems(menu,listenerl) 
.setPositiveButton(" 取 消 "， 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog, int which) 
{ 
} 
BD) 
.Show(); 


6) 定义 openFile(CFile 8 方法 ， 用 于 在 手机 上 打开 指定 的 文件 。 
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private void openFile(File f) 


{ 


Intent intent = new Intent(); 
intent.addFlags(Intent.FLAG ACTIVITY NEW _ TASK); 
intent.setAction(android.content.Intent.ACTION VIEW); 


居 调用 getMIMETypeO 来 取得 文件 类 型 */ 
String type = get MIMEType(D); 

访 设置 intent 的 file 与 MimeType */ 
intent.setDataAndType(Uri.fromFile(f),type); 


j 户 选择 一 个 文件 时 ， 系 统 自 动弹 出 一 个 要 如 何 处 理 文件 的 ListDialog 对 话机 


具体 代码 如 下 。 


TH 
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startActivity(intent); 


} 


7) 定义 方法 getMIMEType(File fj， 用 于 判断 指定 文件 的 类 型 。 具 体 代码 如 下 。 


Je 


判断 文件 MimeType 的 方法 */ 


private String getMIMEType(File f) 


{ 


} 
} 


Wi, 
》 


String type= 

String fName=f.getName(); 

上 # 取得 扩展 名 */ 

String end=fName.substring(fName.lastIndexOf(".")+1, 
fName.length()).toLowerCase(); 


依附 文件 名 的 类 型 决定 MimeType */ 
if(end.equals("m4a")llend.equals("mp3")llend.equals("mid") 
llend.equals("xmf'')llend.equals("ogg")llend.equals("wav")) 


{ 
type = "audio"; 
} 
else if(end.equals("3gp")llend.equals("mp4")) 
{ 
type = "video"; 
} 


else if(end.equals("jpg")llend.equals("gif'')llend.equals("png") 
llend.equals("jpeg")llend.equals("bmp")) 

{ 
type = "image"; 

} 

else 

{ 
旋 如 果 无 法 直接 打开 ， 就 跳出 软件 列表 给 用 户 选择 */ 
type="*"; 

} 


type J We 


return type; 


2. 文件 MyAdapterjava 
接 下 来 讲解 文件 MyAdapterjava 的 具体 实现 代码 。 
1) 分 别 声明 如 下 4 个 变量 。 


口 mIcon1: 回 到 根 目录 的 图 文件 。 
口 mIcon2: 回 到 上 一 层 的 图 文件 。 
口 mIcon3: 文件 夹 的 图 文件 。 
口 mIcon4: 文件 的 图 文件 


o 


画 国 195 


对 前 


196 


Android 开发 完全 实战 宝典 


package irdc.example9; 


/* import 相关 class */ 
import irdc.example9.R; 


import java.io.File; 

import java.util.List; 

import android.content.Context; 
import android.graphics.Bitmap; 


import android.graphics.BitmapFactory; 


import android.view.LayoutInflater; 


import android.view.View; 
import android.view.ViewGroup; 


import android.widget.BaseAdapter; 


import android.widget.ImageView; 


import android.widget. TextView; 


/* 自 定义 的 Adapter， 继 承 于 android.widget.BaseAdapter */ 
public class MyAdapter extends BaseAdapter 


{ 
旋 变量 声明 光 


private LayoutInflater mInflater; 


private Bitmap mlcon!l; 
private Bitmap mlcon2; 
private Bitmap mlcon3; 
private Bitmap mlcon4; 
private List<String> items; 
private List<String> paths; 


2) 定义 MyAdapter 构造 器 ，3 
面 定义 的 4 个 变量 进行 了 赋值 。 


分 别传 入 了 items、paths 和 mInflater 这 3 个 参数 ， 最 后 
号 体 代 码 如 下 。 


庆 MyAdapter 的 构造 器 ， 传 入 3 个 参数 */ 
public MyAdapter(Context context,List<String> it,List<String> pa) 


{ 
/#* 参数 初始 化 */ 


mInfater = LayoutInflater.from(context); 


items = 1t; 


paths = pa; 


ImIconl = BitmapFactory.decodeResource(context.getResources(),R.drawable.back01); 


mlcon2 = BitmapFactory.decodeResource(context.getResources(),R.drawable.back02); 


mlcon3 = BitmapFactory.decodeResource(context.getResources(),R.drawable.folder); 


mlcon4 = BitmapFactory.decodeResource(context.getResources(),R.drawable.doc); 


} 


3) 分 别 覆 新 getCount()、getItem(int position) 和 getItemId(int position)。 具 体 代 码 如 下 。 
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/# 因 继 承 BaseAdapter， 所 以 需要 履 盖 以 下 方法 */ 
(QOverride 
public int getCount() 


{ 


return items.size(); 


} 


(QOverride 
public Object getItem(int position) 


{ 


return items.get(position); 


} 


(QOverride 
public long getItemId(int position) 


{ 


return position; 


} 


4) 使 用 自 定义 的 file_ row 作为 Layout， 然 后 分 别 设置 “ 回 到 根 目 录 ” 的 文字 与 icon 和 
“ 回 到 上 一 层 ” 的 文字 与 icon。 具 体 代码 如 下 。 
(@Override 


public View getView(int position, View convertView,ViewGroup parent) 


{ 
ViewHolder holder; 


if(convertView == null) 
{ 
/* 使 用 自 定 义 的 fle row 作为 Layout */ 
convertView = mInflater.inflate(R.layout.file_row, null); 
/* 初始 化 holder 的 text 与 icon */ 
holder = new ViewHolder(); 
holder.text = (TextView) convertView.find ViewBylId(R.id.text); 
holder.icon = (ImageView) convertView.findViewById(R.id.icon); 


convertView.setTag(holder); 


} 


else 


{ 
holder = (ViewHolder) convertView.getTag(); 


} 


File 人 new File(paths.get(position).toString()); 
/x* 设置 [ 回 到 根 目录 ] 的 文字 与 icon */ 
if(items.get(position).toString().equals("b1")) 

{ 
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执行 后 的 初始 效果 如 图 5-29 所 示 。 当 选择 一 个 文件 夹 后 会 弹出 一 个 操作 对 话 框 ， 


} 


holder.text.setText("Back to /"); 
holdericon.setImageBitmap(mIcon1l); 
} 
/# 设置 [ 回 到 上 一 层 ] 的 文字 与 icon */ 
else if(items.get(position).toString().equals("b2")) 
{ 
holder.text.setText("Back to .."); 
holder.icon.setImageBitmap(mlcon2); 
} 
/* 设置 [文件 或 文件 来] 的 文字 与 icon */ 
else 
{ 
holder.text.setText(f.getName()); 
if(f.isDirectory()) 
{ 
holdericon.setImageBitmap(mIcon3); 
} 
else 
{ 
holdericon.setImageBitmap(mIcon4); 
} 
} 
return convertView; 
} 
/* class ViewHolder */ 
private class ViewHolder 


1 


TextView text; 


ImageView icon; 


} 


可 以 选择 “打开 文件 ”“ 更 改 文件 名 ”和 “删除 文件 ”这 3 种 操作 ， 如 图 5-30 所 示 。 
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当选 择 图 5-30 中 的 一 种 操作 后 ， 会 执行 对 应 的 功能 。 


5.B 00He0 Cachne0 DO 


当 Android 程序 运行 后 ， 可 以 通过 程序 来 获取 正在 运行 程序 的 路 径 ， 此 路 径 通 常 位 于 
“/data/data/package name”， 此 处 的 package name 就 是 程序 的 Package 名 称 。 在 本 节 的 内 容 
中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 获取 File 和 Cache 的 路 径 的 基本 流程 。 本 实例 的 源 代 
码 保存 在 “光盘 :daimaSvexample13”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


5.13.1 ”实现 原理 


在 本 实例 中 ， 首 先 设计 了 两 个 按钮 ， 分 别 触 发 获取 File 和 Cache 路 径 的 事件 。 当 用 户 选 
择 后 ， 会 弹出 一 个 新 页 面 ， 此 新 页 面 以 ListActivity 来 显示 里 面 的 目录 或 文件 。 当 打开 目录 
时 ， 还 可 以 看 到 该 目录 下 的 文件 。 上 述 功 能 是 通过 getCacheDir 方法 和 getFilesDir 方法 实现 
的 ， 两 个 方法 都 能 获得 当前 的 手机 自 带 的 存储 空间 中 的 当前 包 文件 的 路 径 。 


5.13.2 具体 实现 


本 实例 的 主 程序 文件 是 examplel3.java 和 example13 1.java， 下 面 开 始 讲解 其 具体 实现 
代码 。 

1. 文件 example13.java 

下 面 先 讲解 文件 example13.java 的 具体 实现 代码 。 

1) 先 定义 两 个 按钮 变量 myButtonl 和 myButton2， 然 后 定义 两 个 文件 变量 cacheDir 和 
fileDir。 有 具体 代码 如 下 。 


public class example13 extends Activity 


{ 


private Button myButton 1 ; 


private Button myButton2; 
private File cacheDir; 
private File fileDir; 


2) 根据 单 击 的 按钮 ， 分 别 获取 目前 Cache 目录 和 目前 File 目录 。 有 具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


myButtonl = (Button) fndViewById(R.id.myButton1); 
myButton2 = (Button) fndViewById(R.id.myButton2); 


/# 取得 目前 Cache 目录 */ 
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cacheDir = this.getCacheDir(); 
放 取得 目前 File 目录 */ 
fileDir = this.getFilesDir(); 


3) 分 别 定义 两 个 按钮 的 单 击 处 理 
OnClickListener。 上 有 具体 代码 如 下 。 


山中 


有 件 myButton1.setOnClickListener 和 myButton2.set- 


myButtonl.setOnClickListener(new Button.OnClickListener() 


{ 


Override 
public void onClick(View arg0) 


{ 


String path = fileDir.getParent() + java.io.File.separator 
+ fileDir.getName(); 


showListActivity(path); 


} 
); 


myButton2.setOnClickListener(new Button.OnClickListener() 


{ 


(QOverride 
public void onClick(View arg0) 
{ 


String path = cacheDir.getParent() + java.io.File.separator 
+ cacheDir.getName(); 


showListActivity(path); 


); 


4) 调用 example13_ 1， 然后 将 获取 路 径 传 入 。 有 具体 代码 如 下 。 


private void showListActivity(String path) 
{ 


Intent intent = new Intent(); 


intent.setClass(examplel13.this, examplel3 1.class); 


Bundle bundle = new Bundle(); 
bundle.putString("path", path); 
intent.putExtras(bundle); 
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startActivity(intent); 


} 


} 


2. 文件 example13_1.java 
接 下 来 看 文件 example13_1.java 的 具体 实现 代码 。 
1) 先 定 义 类 example13 _ 1， 并 继承 了 ListActivity。 有 具体 代码 如 下 。 


public class example13 1 extends ListActivity 


{ 
private List<String> items = null; 
private String path; 


/** Called when the activity is first created. */ 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.mylist); 


Bundle bunde = this.getIntent().getExtras(); 
/* 取得 example13 所 传 的 路 径 * 
path = bunde.getString("path"); 


this.setTitle(path); 


Java.io.File file = new java.io.File(path); 
谨 列 出 该 路 径 下 的 所 有 文件 汉 
fill(file.listFiles()); 

} 


2) 定义 onListltemClick 方法 ， 用 于 显示 传 入 的 文件 和 文件 目录 。 上 其 体 代码 如 下 。 


(QOverride 
protected void onListItemClick 
(ListView 1 View v, int position, long 1d) 
{ 
File file = new File 
(path + java.io.File.separator + items.get(position)); 


if (file.isDirectory()) 


{ 
fill(file.listFiles()); 


} 
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3) 定义 fl(File[] files) 方 法 ， 将 取得 的 文件 名 放 入 ArrayList。 有 具体 代码 如 下 。 


private void fill(File[] files) 
{ 


items = new ArrayList<String>(); 


if (files == null) 
{ 


return; 


/* 取得 文件 名 放 入 ArrayList */ 


for (File file : files) 


{ 
items.add(file.getName()); 


/* 将 ArrayList 放 入 ArrayAdapter */ 
ArrayAdapter<String> fileList = new ArrayAdapter<String> 
(this, android.R.layout.simple list item 1, items); 


setListAdapter(fileList); 


} 


3. 设置 Activity 


接 下 来 在 文件 AndroidManifest.xml 中 设置 两 个 Activity， 一 个 在 LAUNCHER 启动 时 运 


行 ， 另 一 个 为 取得 文件 夹 信息 时 所 需要 显示 的 Activity。 


具体 代码 如 下 。 


<manifest xmlns:android="http://schemas.android.com/apk/res/android" 


package="irdc.example13" 
android:versionCode="1" 


android:versionName="1.0 


0 


<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 


<activity android:name= 


".examplel13" 


android:label="(@string/app_name"> 


<intent-filter> 


<action android:name="android.intent.action.MAIN" /> 


<category android:name="android.intent.category.LAUNCHER" /> 


</intent-filter> 


</activity> 


<activity android:name="examplel3 1"></activity> 


</application> 


</manifest> 


执行 后 会 显示 两 个 按钮 ， 如 图 


5-31 所 示 。 当 单 避 


! 


0 


个 按钮 后 ， 会 显 


息 。 例 如 ， 单 击 “ 现 在 File” 按 钮 后 的 效果 如 图 5-32 所 示 。 
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在 手机 应 用 中 ，Wi-Fi 也 是 一 个 十 分 重要 的 通信 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具 
体 实例 的 实现 ， 介 绍 在 Android 系统 中 对 Wi-Fi 进行 控制 基本 流程 。 本 实例 的 源 代 码 保存 在 
“光盘 :daimaNS\example14”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


5.14.1 Wi-Fi 简 介 


Wi-Fi 是 一 种 可 以 将 个 人 电脑 、 手 持 设 备 〈 如 PDA、 手 机 ) 等 终端 以 无 线 方式 互相 连接 
的 技术 。Wi-Fi 是 一 个 无 线 网 路 通信 技术 的 品牌 ， 由 Wi-Fi 联盟 (Wi-Fi Alliance) 所 持 有 。 


目的 是 改善 基 呈 


IEEE 802.11 标准 的 无 线 网 路 产品 之 间 的 互通 性 。 现 时 一 般 人 会 把 Wi-Fi 及 


IEEE 802.11 混为一谈 。 甚 至 把 Wi-Fi 等 同 于 无 线 网 际 网 路 。 
Wi-Fi 就 是 一 种 无 线 联 网 的 技术 ， 以 前 通过 网 线 连接 电脑 ， 而 现在 则 是 通过 无 线 电波 来 


连 网 ， 和 常 见 的 就 是 一 个 无 线路 由 器 ， 在 这 个 无 线路 由 器 的 电波 覆盖 的 有 效 范围 都 可 以 条 


Wi-Fi 连接 方式 进行 联网 ， 如 果 无 线路 由 器 连接 了 一 条 ADSL 线路 或 者 别 的 上 网 线路 ， 则 又 


被 称 为 “热点 ”。 


现在 市 面 上 常见 的 无 线路 由 器 多 为 54M 速度 ， 再 上 一 个 等 级 就 是 108M 的 速度 ， 当 然 


这 个 速度 并 不 是 你 访问 互联 网 的 速度 ， 访 问 互 联网 的 速度 主要 是 取决 于 Wi-Fi“ 热 点 ”的 互 
联网 线路 。Wi-Fi 是 一 种 帮助 用 户 访问 电子 邮件 、Web 和 流 式 媒体 的 技术 。 它 为 用 户 提供 了 
无 线 的 宽带 互联 网 访问 。 同 时 ， 它 也 是 在 家 里 、 办 公 室 或 在 旅途 中 上 网 的 快速 、 便 捷 的 途 
径 。 能 够 访问 Wi-Fi 网 络 的 地 方 被 称 为 “热点 ”。 

Wi-Fi 热点 是 通过 在 互联 网 连接 上 安装 访问 点 来 创建 的 。 


5.14.2 ”实现 原理 


在 Android 系统 中 ， 存 在 一 个 无 线 控制 模块 ， 打 开 方 式 如 下 。 依 次 单 击 “Menu” 一 
“Settings” 一 “Wireless$networks” 一 “Wi-Fi settings” 出 现 图 5-33 所 示 界 面 。 
图 5-33 所 示 的 是 Wi-Fi 的 控制 界面 ， 可 以 控制 Wi-Fi 的 打开 和 关闭 ， 而 本 实例 的 目的 是 


以 编程 的 方式 实现 类 似 的 功能 。 
在 本 实例 中 ， 一 共用 到 了 WifiManager 的 五 种 状态 ， 有 具体 说 明 如 下 。 


口 WifiManagerWIFI STATE ENABLING: 表示 Wi-Fi 已 经 打开 。 
口 WifiManager.WIFI STATE DISABLING: 表示 Wi-Fi 正在 关闭 而 无 法 关闭 。 
口 WifiManager.WIFI STATE DISABLED: 表示 Wi-Fi 已 经 关闭 。 
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图 5-33 “WiFi 控制 界面 


口 WifiManager.WIFI STATE ENABLING: 表示 Wi-Fi 正在 打开 而 无 法 关闭 。 
口 WifiManager.WIFI STATE ENABLED: 表示 Wi-Fi 已 经 打开 无 法 再 打开 。 
口 WifiManager.WIFI STATE DISABLING: 表示 Wi-Fi 正在 关闭 。 

口 WifiManager.WIFI STATE DISABLED: 表示 Wi-Fi 已 经 关闭 。 

口 WifManagerWIFI STATE UNKNOWN: 表示 Wi-Fi 无 法 识别 。 

在 具体 实现 上 ， 会 首先 定义 一 个 复 选 框 CheckBox， 然 后 捕捉 CheckBox 的 单 击 事件 ， 根 
据 对 应 的 状态 显示 对 应 的 提示 。 例 如 ， 可 以 用 下 面 的 代码 检测 了 Wi-Fi 是 否 启动 。 

WifiManager wm = (WifiManager) context.getSystemService(Context. WIFI SERVICE); 


if(wm.getWifiState() == WifiManager. WIFI STATE ENABLED){ 
return true; 


} 


5.14.3 ”具体 实现 


本 实例 的 主 程序 文件 是 examplel4.java， 下 面 开始 讲解 其 具体 实现 代码 。 
1) 引入 wifi.WifiManager 和 widget.CheckBox， 然 后 创建 WiFiManager 对 象 mWiFi 
Manager01。 具 体 代码 如 下 。 


public class example14 extends Activity 
{ 
private TextView mTIextView01; 
private CheckBox mCheckBox01; 


/* 创建 WiFiManager 对 象 */ 
private WifiManager mWiFiManager01; 


2) 分 别 定义 两 个 变量 : mTextView01 和 mCheckBox01， 分 别 用 于 显示 提示 文本 和 获取 
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复 选 框 的 选择 状态 。 有 具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mTlextView01 = (TextView) fndViewById(R.id.myTextView1l); 
mCheckBox01 = (CheckBox) findViewById(R.id.myCheckBox1); 


3) 以 getSystemService( ) 方 法 取得 WIFI SERVICE (WIFI 服务 )， 具 体 代码 如 下 。 


mWiFiManager01 = (WifiManager) 
this.getSystemService(Context.WIFI SERVICE); 


4) 通过 让 语句 来 判断 运行 程序 后 的 WiFi 状态 是 否 打开 或 打开 中 ， 这 样 便 可 显示 对 应 的 
提示 信息 。 具 体 代码 如 下 。 


雍 判断 运行 程序 后 的 WiFi 状态 是 否 打开 或 打开 中 */ 
if(mWiFiManager01.isWifiEnabled()) 


{ 


上 # 判断 WiFi 状态 是 否 “ 已 打开 ” */ 
if(mWiFiManager01.getWifiState()=— 
WifiManager. WIFI STATE ENABLED) 
{ 
雍 若 WiFi 已 打开 ， 将 选取 项 打 勾 六 
mCheckBox01.setChecked(true); 
雍 更 改选 取 项 文字 为 关闭 WiFi*/ 
ImCheckBox01.setIext(R.string.str_ uncheck); 
} 
else 
{ 
族 若 WiFi 示 打开， 将 选取 项 勾 选取 消 光 
mCheckBox01.setChecked(false); 
久 更 改选 取 项 文字 为 打开 WiFi*/ 
ImCheckBox01.setIext(R.string.str_ checked); 
} 
) 
else 
{ 
mCheckBox01.setChecked(false); 
mCheckBox01.setText(R.string.str checked); 
} 


5) 通过 mCheckBox01.setOnClickListener 来 捕捉 CheckBox 的 单 击 寻 


| 
mul 
pr 


EF， 用 onClick 
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(View 下) 方法 获取 用 户 的 单 击 ， 然 后 使 用 主 语句 ， 根 据 操作 需求 来 执行 对 应 操作 ， 
要 输出 对 应 的 提示 信息 。 有 具体 代码 如 下 。 


湾 
由 
准 


mCheckBox01.setOnClickListener( 
new CheckBox.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 

{ 


雍 当选 取 项 为 取消 选取 状态 */ 
if(mCheckBox01.isChecked()==false) 


{ 
上 # 尝试 关闭 WiFi 服务 */ 
try 
{ 


谨 判断 WiFi 状态 是 否 为 已 打开 党 
if(mWiFiManager01.isWifiEnabled() ) 
{ 
从 关闭 WiFi *%/ 
if{mWiFiManager01.setWifiEnabled(false)) 
{ 
ImTIextView01.setIext(R.string.str_ stop wifi done); 
} 
else 
{ 
mTlextView01.setText(R.string.str_ stop wifi failed); 
} 
} 
else 
{ 
诺 WiFi 状态 不 为 已 打开 状态 时 党 
switch(mWiFiManager01.getWifiState()) 
{ 


人 #WiFi 正在 打开 过 程 中 ， 导 致 无 法 关闭 ... */ 
case WifiManager. WIFI STATE ENABLING: 
mTextView01.setText 
( 
getResources().getText 
(R.string.str_ stop wifi failed)+":"+ 


getResources().getText 
(R.string.str wifi enabling) 
); 
break; 
/WiFi 正在 关闭 过 程 中 ， 导 致 无 法 关闭 … */ 
case WifiManager.WIFI STATE DISABLING: 


| 
| 
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mTlextView01.setText 
( 
getResources().getText 
(R.string.str_stop_ wifi failed)+t":"+ 
getResources().getText 
(R.string.str wifi disabling) 
); 
break:; 
族 WiFi 已 经 关闭 */ 
case WifiManager.WIFI STATE DISABLED: 
mTlextView01.setText 


getResources().getText 
(R.string.str_ stop_ wifi failed)+":"+ 
getResources().getText 
(R.string.str wifi disabled) 
); 
break; 
放 无 法 取得 或 辨识 WiFi 状态 * 
case WifiManager.WIFI STATE UNKNOWN: 
default: 
mTextView01.setText 
( 
getResources().getText 
(R.string.str_stop wifi failed)+t":"+ 
getResources().getText 
(R.string.str wifi unknow) 
); 
break; 


} 
mCheckBox01.setText(R.string.str checked); 


} 


catch (Exception e) 

{ 
Log.i("HIPPO", ¢.toString()); 
e.printStackTrace(); 


} 
else if(mCheckBox01.isChecked()==true) 


{ 
此 尝试 打开 WiFi 服务 */ 
try 
{ 
/# 确认 WiFi 服务 是 关闭 且 不 在 打开 作业 中 */ 


手机 交互 应 用 服务 “二 
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if(!ImWiFiManager01.isWifiEnabled() && 
mWiFiManager01.getWifiState()!= 
WifiManager. WIFI STATE ENABLING ) 


{ 
if{mWiFiManager01.setWifiEnabled(true)) 


\ 
switch(m WiFiManager01.getWifiState()) 


{ 


上 六 WiFi 正在 打开 过 程 中 ， 导 致 无 法 打开 ... */ 
case WifiManager. WIFI STATE ENABLING: 
mTextView01.setText 


( 


getResources().getText 
(R.string.str wifi enabling) 
); 
break; 
旋 WiFi 已 经 为 打开 ， 无 法 再 次 打开 … */ 
case WifiManager. WIFI STATE _ ENABLED: 
ImTextView01.setIext 


( 


getResources().getText 
(R.string.str start wifi done) 
); 
break; 
其 他 未 知 的 错误 */ 
default: 
mTextView01.setText 


( 


getResources().getText 
(R.string.str_start wifi failed)+":"+ 
getResources().getText 
(R.string.str wifi unknow) 

); 

break; 


} 


else 


{ 


mTextViewO1.setText(R.string.str_start wifi failed); 


} 


else 


{ 
switch(mWiFiManager01.getWifiState()) 


{ 


t 


六 WiEi 正在 打开 过 程 中 ， 导 至 无 法 打开 .. 


中/ 
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case WifiManager.WIFI STATE ENABLING: 
mTextView01.setText 
( 
getResources().getText 
(R.string.str start wifi failed)+":"+ 
getResources().getText 
(R.string.str wifi enabling) 
); 
break; 
作 WiFi 正在 关闭 过 程 中 ， 导 致 无 法 打开 ... */ 
case WifiManager.WIFI STATE DISABLING: 
mTextView01.setText 
( 
getResources().getText 
(R.string.str start wifi failed)+":"+ 
getResources().getText 


(R.string.str wifi disabling) 
); 
break; 
访 WiFi 已 经 关闭 */ 
case WifiManager.WIFI STATE DISABLED: 
mTlextView01.setText 
( 
getResources().getText 
(R.string.str start wifi failed)+":"+ 
getResources().getText 
(R.string.str wifi disabled) 
); 
break; 
庶 无 法 取得 或 识别 WiFi 状态 汶 
case WifiManager.WIFI STATE UNKNOWN: 
default: 
mTlextView01.setText 
( 
getResources().getText 
(R.string.str_start wifi failed)+":"+ 
getResources().getText 
(R.string.str wifi unknow) 
); 
break; 


} 


mCheckBox01.setText(R.string.str_uncheck); 


} 


catch (Exception e) 


{ 
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Log.i("HIPPO", e.toString(O); 
e.printStackTrace(); 


); 
} 


6) 定义 mMakeTextToast(String str, boolean isLong)，| 


信息 。 有 具体 代码 如 下 。 


public void mMakeTextToast(String str, boolean isLong) 


{ 
if(isLong==true) 
{ 
Toast.makeText(example14.this, str, Toast.LENGTH LONG).show(); 
} 
else 
{ 
Toast.makeText(example14.this, str, Toast.LENGTH SHORT).show(); 
} 
} 
(@Override 
protected void onResume() 
{ 


/TODO Auto-generated method stub 


/* 在 onResume 重 写 事件 为 取得 打开 程序 当下 Wi-Fi 的 状态 */ 


try 
{ 
switch(mWiFiManager01.getWifiState()) 
{ 
上 WiFi 已 经 在 打开 状态 … */ 
case WifiManager. WIFI STATE ENABLED: 
mTextView01.setText 


( 


getResources().getText(R.string.str wifi enabling) 


》» 
break; 

/WiFi 正在 打开 过 程 中 状态 … */ 

case WifiManager. WIFI STATE ENABLING: 
mTlextView01.setText 


( 


getResources().getText(R.string.str wifi enabling) 


D); 
break:; 
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接 下 来 需要 在 文件 AndroidManifest.xml 


入 WiFi 正在 关闭 过 程 中 … */ 
case WifiManager. WIFI STATE DISABLING: 
ImIextView01.SetIext 


( 
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getResources().getText(R.string.str wifi _ disabling) 


); 
break; 

/WiFi 已 经 关闭 */ 

case WifiManager. WIFI STATE DISABLED: 
mTlextView01.setText 


( 


getResources().getText(R.string.str wifi disabled) 


); 

break; 
启 无 法 取得 或 识别 Wi-Fi 状态 六 
case WifiManager.WIFI STATE UNKNOWN: 
default: 

mTlextView01.setText 


( 


getResources().getText(R.string.str _ wifi unknow) 


); 
break; 


} 


catch(Exception e) 


{ 


mfTlextView01.setText(e.toString()); 
e.getStack Trace(); 
} 


super.onResume(); 


(QOverride 
protected void onPause() 


{ 


super.onPause(); 


} 


具体 代码 如 下 。 


<manifest xmlns:android="http://schemas.android.com/apk/res/android" package: 


android:versionCode="1" android:version Name="1.0.0"> 
- <application android:icon="(@drawable/icon" android:label 


Ny 二 1t 


- <activity android:name: 
- <intent-filter> 


.example14" android:label 


ee 


@string/app_name"> 


@string/app_name"> 


手机 交互 应 用 服务 


添加 对 Wi-Fi 的 访问 以 及 对 网 络 状态 的 权限 。 


irdc.examplel14" 
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<action android:name="android.intent.action.MAIN" /> 
<category android:name="android.intent.category.LAUNCHER" /> 
</intent-filter> 
</activity> 
</application> 

<!-- ”新 增 存 取 Wi-Fi 以 及 网 络 等 相关 权限 

本 二 
<uses-permission android:name="android.permission.CHANGE NETWORK STATE" /> 
<uses-permission android:name="android.permission.CHANGE WIFI STATE"/> 
<uses-permission android:name="android.permission.ACCESS NETWORK STATE'" /> 
<uses-permission android:name="android.permission.ACCESS WIFI STATE'" /> 
<uses-permission android:name="android.permission.INTERNET" /> 


<uses-permission android:name="android.permission.WAKE LOCK"/> 
</manifest> 
执行 后 会 显示 两 个 按钮 ， 如 图 5-34 所 示 。 当 选择 复 选 框 后 会 执行 对 应 的 操作 处 理 ， 并 
且 显 示 对 应 的 提示 信息 ， 如 图 5-35 所 示 。 


> 


一 动 硬 大 3:44m 痪 天 全 3:45 mw 
example14 example14 
已 天 闭 打开 失败 :未 知 … 
国 17. (MA.… 
图 5-34 ”初始 效果 图 5-35 ”文件 信息 
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在 手机 应 用 中 ， 通 常 需要 获取 SIM 卡 内 的 信息 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实 
例 的 实现 ， 介 绍 在 Android 系统 中 获取 SIM 卡 内 信息 的 基本 流程 。 本 实例 的 源 代码 保存 在 
“光盘 :\daima\5S\example15” 下 面 开 始 讲 解 本 实例 的 具体 实现 流程 。 


5.15.1 SIM 卡 简介 


SIM 卡 是 客户 识别 模块 (Subscriber Identity Module，SIM) 的 缩写 ， 也 称 为 智能 卡 、 用 
户 身 份 识别 卡 ，GSM 数字 移动 电话 机 必须 装 上 此 卡 方 能 使 用 。 它 在 电脑 芯片 上 存储 了 数字 
移动 电话 客户 的 信息 ， 加 密 的 密 钥 等 内 容 ， 可 供 GSM 对 网 络 客户 身份 进行 鉴别 ， 并 对 客户 
通话 时 的 语音 信息 进行 加 密 。SIM 卡 的 使 用 ， 完 全 防止 了 并 机 和 通话 被 窍 听 行 为 ,并且 SIM 
卡 的 制作 是 严格 按照 GSM 国际 标准 和 规范 来 完成 的 ， 从 而 可 靠 的 保障 了 客户 的 正常 通信 。 

为 防止 他 人 擅 用 您 的 SIM 卡 ， SIM 卡 设置 了 个 人 识别 密码 一 一 PIN 码 ， 只 要 设置 了 
PIN 码 用 户 在 每 次 打开 手机 时 ， 屏 幕 上 会 显示 要 求 输入 4 位 PIN 码 ， 初 始 的 PIN 码 为 1234 
或 0000， 连 续 输入 三 次 错误 码 的 PIN 码 ， 手 机 会 显示 “输入 PUK 码 ” 或 “已 锁 ” 字 样 ， 说 
明 SIM 卡 已 被 锁 上 ， 这 时 千 万 不 要 再 按 手 机 键盘 ， 因 为 如 果 输 入 的 PIN 解锁 码 累 计 超 过 10 
次 ，SIM 卡 将 自动 报废 。 当 SIM 卡 被 锁 时 ， 需 携带 SIM 卡 身 份 证 到 移动 营业 厅 ， 由 服务 人 
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5.15.2 ”实现 原理 


在 本 实例 中 ， 
Manager) 对 象 提 


通过 Android API 中 的 TelephonyManager 〈 即 Android.telephony. Telephony 
供 的 几 个 方法 实现 对 SIM 卡 信息 的 读 取 。 另 外 ，TelephonyManager 还 能 获 


取 手 机 的 号 码 。 看 下 面 的 代码 。 


TelephonyManager tm = (TelephonyManager)this.getSystemService(Context.TELEPHONY _ SERVICE); 
numberText.setText(tm.getLinel Number()); 


其 中 ， 上 面 加 粗 部 分 能 够 获取 本 机 号 码 ， 除 此 之 外 ，TelephonyManager 〈 手 机 通信 管 


理 ) 类 还 提供 了 多 种 获取 手机 信息 的 函数 ， 例 如 imei()、imsi() 等 。 看 下 面 的 代码 。 


package com.pingan.innovation; 


import android.app.Activity; 


import android.content.Context; 


import android.os.Bundle; 


import android.telephony.TelephonyManager; 


import android.widget. TextView; 


public class PhoneInfo extends Activity { 


private TextView numberText; 


private TextView imeiText; 


private TextView onText; 


private TextView snText; 


private TextView imsiText; 


private TextView ssText; 


private TextView ntText; 


(QOverride 
public void onCreate(Bundle savedInstanceState) { 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


numberText = (TextView) findViewById(R.id.numberText); 

imeiText = (TextView) findViewByld(R.id.imeiText); 

onText= (TextView) findViewById(R.id.onText); 

snText = (TextView) findViewById(R.id.snText); 

imsiText = (TextView) findViewByld(R.id.imsiText); 

ssText = (TextView) findViewByld(R.id.ssText); 

ntText = (TextView) findViewByld(R.id.ntText); 

TelephonyManager tm = (TelephonyManager)this.getSystemService(Context.TELEPHONY _ 
SERVICE); 

numberText.setText(tm.getLinel Number()); 
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imeiText.setText(tm.getDeviceld()); 
onText.setText(tm.getNetworkOperatorName()); 
snText.setText(tm.getSimSerialNumber()); 
imsiText.setText(tm.getSubscriberld()); 
ssText.setText(tm.getNetworkCountryIso()); 
ntText.setText(tm.getNetworkOperator()); 


} 


通过 上 述 代 码 中 的 几 个 函数 ， 分 别 获取 了 手机 号 码 、imei、 运 营 商 名 称 、sim 卡 序列 
号 、IMSI、sim 卡 所 在 国家 以 及 运营 商 编 号 。 运 行 效果 如 图 5-36 所 示 。 


手机 号 码 : 15555218135 
IMEI : 000000000000000 


运营 商 : Android 


sim 卡 序列 号 : 89014103211118510720 


IMSI : 310260000000000 


ISO : US 


运营 商 编 号 : 310260 


5.15.3 ”具体 实现 


本 实例 的 主 程序 文件 是 example15.java 和 MyAdapter.java， 下 面 开始 讲解 其 具体 实现 
代码 。 

1. 文件 example15.java 

下 面 先 讲解 文件 example15.java 的 具体 实现 代码 。 

1) 先 载 入 布局 文件 main.xml， 然 后 通过 add(getResources().getText(R.string.str_list0). 
toStringO) 将 取得 的 信息 写 入 List〈 列 表 ) 中， 最 后 通过 if 语句 来 设置 SIM 卡 的 状态 。 有 具体 
代码 如 下 。 


public class example15 extends ListActivity 
private TelephonyManager telMer; 
private List<String> item=new ArrayList<String>(); 
private List<String> value=new ArrayList<String>(); 


@Suppress Warnings("static-access") 
@Override 
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第 5 章 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/# 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


手机 交互 应 用 服务 


telMgr = (TelephonyManager)getSystemService( TELEPHONY SERVICE); 


放 将 取得 的 信息 写 入 List 中 * 
举 取得 SIM 卡 状 态 */ 
item.add(getResources().getText(R.string.str_list0).toString()); 
if(telIMegr.getSimState()=—=tel Mgr.SIM STATE READY) 
{ 
value.add(" 恨 好 "); 
} 
else if(telI Mgr.getSimState()=—tel Mgr.SIM_ STATE ABSENT) 
{ 
value.add(" 无 SIM 卡 "); 
} 
else 
{ 
value.add("SIM 卡 被 锁定 或 未 知 的 状态 "); 
} 


7IT 


后 使 ) 


item.add(getResources().getText(R.string.str list1).toString()); 
if(telI Mgr.getSimSerialNumber()!=null) 
{ 
value.add(tel Megr.getSimSerialNumber()); 
} 
else 
{ 
value.add(" 无 法 取得 "); 
} 


/* 取得 SIM 卡 供 货 商 代码 */ 
item.add(getResources().getText(R.string.str_list2).toString()); 
if(telI Mgr.getSimOperator().equals("")) 
{ 
Value.add(" 无 法 取得 "); 
} 
else 
{ 
value.add(tel Megr.getSimOperator()); 
} 


号 、SIM 卡 供 货 商 代码 、SIM 卡 供 货 商 名 称 及 SIM 1 
自 定义 的 MyAdapter 来 将 数据 传 入 ListActivity。 有 具体 代码 如 下 。 


ZL 


el 
沽 
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旋 取得 SIM 卡 供 货 商 名 称 * 
item.add(getResources().getText(R.string.str_ list3).toString()); 


if(telMgr.getSimOperatorName().equals("")) 
{ 

value.add(" 无 法 取得 "); 
} 
else 
\ 

value.add(tel Mgr.getSimOperatorName()); 
} 


从 取得 SIM 卡 国 别 * 
item.add(getResources().getText(R.string.str_list4).toString()); 
if(telIMgr.getSimCountryIso().equals("")) 
{ 
value.add(" 无 法 取得 "); 
} 
else 
{ 
value.add(tel Megr.getSimCountryIso()); 
} 


/* 使 用 自 定义 的 MyAdapter 来 将 数据 传 入 ListActivity */ 
setListAdapter(new MyAdapter(this,item,value)); 


} 


2. 文件 MyAdapterjava 
接 下 来 讲解 文件 MyAdapterjava 的 具体 实现 代码 。 
1) 先 声明 3 个 变量 mInflater、items 和 values， 然 后 定义 MyAdapter 构造 器 ， 传 入 三 个 
参数 。 有 具体 代码 如 下 。 
/* 自 定义 的 Adapter， 继 承 android.widget.BaseAdapter */ 


public class MyAdapter extends BaseAdapter 


{ 
族 变量 声明 光 
private LayoutInflater mInflater; 


private List<String> items; 

private List<String> values; 

/* MyAdapter 的 构造 器 ， 传 入 三 个 参数 *%/ 

public MyAdapter(Context contexbList<String> item, 

List<String> value) 

{ 
入 参数 初始 化 忆 
mInfater=LayoutInflater.from(context); 


items = item; 
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values = value; 
} 
2) 分 别 覆 盖 方 法 getCount()、getItem(int position)、getItemId(int position) 及 getView(int 
position, View convertView，ViewGroup pan， 上 有 具体 代码 如 下 。 


/# 因 继 承 BaseAdapter， 需 覆盖 以 下 方法 */ 
(@Override 
public int getCount() 
{ 
return items.size(); 


} 


(@Override 
public Object getItem(int position) 
{ 

return items.get(position); 


} 


(@Override 
public long getItemld(int position) 
{ 

return position; 


} 


(QOverride 
public View getView(int position, View convertView,ViewGroup par) 


{ 
ViewHolder holder; 


if(convertView == null) 

{ 
/* 使 用 自 定义 的 fle row 作为 Layout */ 
convertView = mInflater.inflate(R.layout.row_layout,null); 
/* 初始 化 holder 的 text 与 icon */ 
holder = new ViewHolder(); 
holdertext1=(TIextView)convertView.findViewById(R.id.myTextl); 
holdertext2=(TIextView)convertView.findViewById(R.id.myText2); 


convertView.setTag(holder); 


} 


else 
i 
holder = (ViewHolder) convertView.getTag(); 
} 
此 设置 要 显示 的 信息 */ 


holder.textl1.setText(items.get(position).toString()); 
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holder.text2.setText(values.get(position).toString()); 


return convertView; 


} 


/* class ViewHolder */ 
private class ViewHolder 
{ 
/* text1: 信息 名 称 
* text2: 信息 内 容 */ 
TextView textl; 
TextView text2; 


》 
最 后 ， 还 需要 在 文件 AndroidManifest.xml 中 设置 读 取 电 话 状态 的 权限 ， 有 具体 代码 如 下 。 


<!-- 设置 READ PHONE STATE 权限 --> 
<uses-permission android:name="android.permission.READ PHONE STATE"></uses-permission> 


执行 后 会 显示 对 应 的 获取 信息 ， 如 图 5-37 所 示 。 


Ny 
89014103211118510720 (你 Cy A 
310260 [A [ean ©) fo | 


Androld 


3 围 国 力图 国 国 国 国 国 国 
QIw le lk IT ly lu li lo le | 


图 5-37 ”执行 效果 


5S-10 国 二 加 


在 触摸 屏 手机 应 用 中 ， 通 常 需 要 触摸 一 个 按钮 来 实现 拨号 处 理 。 在 本 节 的 内 容 中 ， 将 通 
过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 实现 类 似 触摸 拨号 按钮 的 基本 流程 。 本 实例 的 源 


代码 保存 在 “光盘 :\daima\5S\example16”， 下面 开始 讲解 本 实例 的 具体 实现 流程 。 
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5.16.1 ”实现 原理 


在 本 实例 中 ， 通 过 Intent 方式 将 电话 号 码 传递 给 内 置 的 拨号 程序 ， 然 后 内 
现 拨号 处 理 操作 。 利 用 了 startActivity0 方 法 将 程序 焦点 交 给 内 置 的 拨号 程序 
Activity 会 成 为 失去 焦点 ， 并 且 还 会 发 生 onPause0 事 件 ， 直 到 关闭 拨号 程序 ， 

原来 的 Activity。 


5.16.2 具体 实现 


在 具体 实现 上 ， 先 插入 了 一 个 按钮 ， 当 单 击 按钮 后 会 调用 手机 内 置 的 默认 拨号 界面 。 


用 服务 “于 


置 投 号 程序 实 


焦点 也 交还 给 


本 实例 的 主 程序 文件 是 example16.java， 主 要 功能 是 当 用 户 单 击 按钮 后 
intent.action.CALL_BUTTON 调用 默认 的 拨号 界面 。 具 体 代 码 如 下 。 


public class example16 extends Activity 


{ 


private ImageButton myImageButton; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 
super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


myImageButton = (ImageButton) fmndViewById(R.id.myImageButton); 


mylmageButton.setOnClickListener(new ImageButton.OnClickListener() 


{ 


@Override 
public void onClick(View V) 
{ 
从 调用 拨号 的 画面 */ 
Intent myIntentDial = new Intent("android.intent.action.CALL _ BUTTON ); 


startActivity(myIntentDial); 


执行 后 会 显示 对 应 的 按钮 ， 如 图 5-38 所 示 。 单 击 按钮 后 ， 会 自动 来 到 系 
拨号 界面 ， 如 图 5-39 所 示 。 


通过 android. 


统 内 置 的 默认 
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碟 畴 售 4:04rm 
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丽 


5-38 ”初始 效果 图 5-39 内置 的 拨号 界面 
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电脑 中 的 任务 管理 器 大 家 都 不 会 陌生 ， 其 中 的 进程 管理 器 能 够 显示 当前 正在 运行 的 程 
序 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 查看 Android 中 正在 运行 程序 的 
基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\5\example17”， 下 面 开始 讲解 本 实例 的 具体 
实现 流程 。 


5.17.1 ”实现 原理 
在 本 实例 中 ， 首 先 插入 了 一 个 按钮 ， 当 单 击 按钮 后 会 显示 当前 正在 运行 的 程序 。 当 前 运 
行程 序 是 通过 ActivityManagergetRunningTasks( ) 方 法 获取 的 ， 然 后 通过 ListView 将 获取 的 信 
县 显示 出 来 。 

当 单 击 按钮 后 ， 如 果 在 ListView 的 工作 已 经 结束 或 被 操作 系统 回收 ， 则 是 不 会 更 新 运行 
列表 的 。 另 外 ， 如 果 不 具 有 访问 其 他 运行 程序 的 权限 ， 也 不 会 显示 在 ListView 列表 。 


注意 ; 为 了 保证 Android 程序 的 高 效 运 行 ， 建 议 限 制 获取 运行 程序 的 数量 ， 在 本 实例 中 
设置 了 最 多 获取 30 个 进程 。 


5.17.2 具体 实现 
本 实例 的 主 程序 文件 是 example17.java， 下 面 开始 讲解 其 具体 实现 流程 。 
1) 设置 类 成 员 最 多 能 够 获取 30 个 Task， 具 体 代码 如 下 。 
public class example17 extends Activity 
{ 
private Button mButton01; 
private ListView mListView01; 
private ArrayA dapter<String> aryA dapterl; 
private ArrayList<String> arylistTask; 


旋 类 成 员 设 置 最 多 几 笔 的 Task 数量 */ 
private int intGetTastCounter=30; 
2) 设置 类 成 员 ActivityManager 的 对 象 ， 然 后 定义 当 单 击 按钮 后 取得 正在 后 台 运 行 的 工 
作 程序 。 有 具体 代码 如 下 。 
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举 类 成 员 ActivityManager 对 象 */ 
private ActivityManager mActivityManager; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButton01 = (Button)findViewByld(R.id.myButton!1); 
mListView01 = (ListView)findViewById(R.id.myListView!1); 


席 单 击 按钮 取得 正在 后 台 运 行 的 工作 程序 沁 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
// TODO Auto-generated method stub 
try 
{ 
/#ActivityManager 对 象 向 系统 取得 ACTIVITY_SERVICE */ 
mActivityManager = (ActivityManager) 
examplel17.this.getSystemService(ACTIVITY SERVICE); 


arylistTask = new ArrayList<String>(); 


启 以 getRunningTasks 方法 取 回 正在 运行 中 的 程序 TaskInfo */ 
List<ActivityManager. RunningTaskInfo> mRunningTasks = 


mActivityManager.getRunning Tasks(intGetTastCounter); 


inti= 1; 

人/# 以 循环 及 baseActivity 方式 取得 工作 名 称 与 ID */ 

for (ActivityManagerRunningTaskImfo amTask : mRunningTasks) 

{ 
/#baseActivity getClassName 取出 运行 工作 名 称 * 
arylistTask.add("™" + (i++) + ": "+ 
amTask.baseActivity.getClassName()+ 
"(ID=" + amTask.id +")"); 

} 

aryAdapterl =new ArrayAdapter<String> 


(example17.this, R.layout.simple list item 1, arylistTask); 


这 aryAdapterl.getCountO==0) 
{ 
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和 # 当 没 有 任何 运行 的 工作 ， 则 提示 信息 * 
mMakeTextToast 
( 


getResources().getText 


(R.string.str err no running task).toString(), 
true 
); 
} 
else 
{ 
诺 发 现 后 台 运 行 的 工作 程序 ， 以 ListView Widget 条 列 星 现 */ 
ImListView01.setAdapter(aryAdapter1l); 


》 
} 
catch(SecurityException e) 
{ 
访 当 无 GET TASKS 权限 时 (SecurityException 异常 ) 提 示 信 息 */ 
mMakeTextToast 
( 


getResources().getText 
(R.string.str_err permission).toString(), 


3) 设置 用 户 在 运行 工作 选择 时 的 事件 处 理 ， 上 其 体 代 码 如 下 。 


mListView01.setOnItemSelectedListener 
(new ListView.OnItemSelectedListener() 
{ 
(QOverride 
public void onItemSelected 
(AdapterView<?> parent, View v, int id, long arg3) 
{ 
旋 由 于 将 运行 工作 以 数组 存放 ， 所 以 使 用 id 取出 数组 元 素 名 称 党 
mMakeTextToast(arylistTask.get(id).toString(),false); 
} 


Override 
public void onNothingSelected( AdapterView<?> arg0) 


全 
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4) 设置 当 用 户 在 运行 工作 单 击 时 的 事件 处 理 ， 具 体 代码 如 下 。 


率 当 用 户 在 运行 工作 上 单 击 时 的 事件 处 理 汐 
mListView01.setOnItemClickListener 
(new ListView.OnItemClickListener() 
{ 
@Override 
public void onItemClick 


(AdapterView<?> parent, View v, int id, long arg3) 

{ 
上 # 由 于 将 运行 工作 以 数组 存放 ， 故 以 id 取出 数组 元 素 名 称 */ 
mMakeTextToast(arylistTask.get(id).toString(), false); 


} 
); 
} 
5) 定义 mMakeTextToast(String str, boolean isLong) 方 法 ， 用 于 实现 提醒 。 具 体 代码 
如 下 。 
public void mMakeTextToast(String str, boolean isLong) 
{ 
if(isLong==true) 
{ 
Toast.makeText(example17 .this, str, Toast.LENGTH LONG).show(); 
} 
else 
{ 
Toast.makeText(example17 .this, str, Toast.LENGTH SHORT).show(); 
} 
} 


} 


最 后 还 需要 在 文件 AndroidManifest.xml 中 设置 权限 ， 具 体 代码 如 下 。 


<uses-permission android:name="android.permission.GET TASKS"></uses-permission> 


执行 后 会 显示 对 应 的 按钮 ， 如 图 5-40 所 示 。 单 击 “ 正 在 获取 运行 的 程序 ”按钮 后 ， 会 
显示 当前 正在 运行 的 程序 ， 如 图 5-41 所 示 。 


屿 间 但 4:06 pM 轧 轩 但 4:07 pM 
en bed 正在 获取 运行 的 程序 
正在 获取 运行 的 程序 1: irdc.example17.example17(ID=3) 


2: com.android.launcher2.Launcher(ID=2) 
正在 获取 运行 的 程序 


图 5-40 ”初始 效果 图 5-41 当前 运行 程序 
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5.8B 口 日 00D0U 


屏幕 旋转 对 广大 读者 来 说 应 该 不 会 耳 生 ， 很 多 智能 手机 都 能 根据 用 户 拿手 机 的 方式 ， 动 
态 的 模 向 或 纵向 显示 屏幕 中 的 内 容 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 实 
现 更 改 手机 屏幕 方向 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 +daimasvexamplel8”， 下 面 
开始 讲解 本 实例 的 具体 实现 流程 。 


| 


5.18.1 实现 原理 
在 本 实例 中 ， 首 先 插入 了 一 个 按钮 ， 当 单 击 按钮 后 会 先 判断 当前 屏幕 的 方向 ， 即 如 果 是 
横向 显示 则 改 为 纵向 显示 ， 如 果 是 纵向 显示 则 改 为 横向 显示 。 

在 具体 实现 上 ， 如 果 要 改变 屏幕 方向 ， 则 必须 要 覆盖 setRequestedOrientation() 方 法 。 如 
果 要 获取 当前 的 屏幕 方向 ， 则 需要 访问 getRequestedOrientation() 方 法 。 


上 


5.18.2 具体 实现 
本 实例 的 主 程序 文件 是 examplel18.java， 下 面 开始 讲解 其 具体 实现 流程 。 
1) 在 onCreate 方法 中 判断 getRequestedOrientation0 是 否 为 -1， 如 果 是 -1 则 表示 在 
Activity 属性 中 没有 设置 Android:screenOrientation (固定 屏幕 显示 模式 ) 的 值 ， 即 表示 即使 
单 击 了 按钮 ， 也 无 法 判断 出 屏幕 的 方向 ， 不 会 实现 屏幕 方向 更 改 。 有 具体 代码 如 下 。 
public class example18 extends Activity 


{ 


private TextView mTIextView01; 


private Button mButton01; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButton01 = (Button)findViewById(R.id.myButton!); 
mTlextView01 = (TextView)findViewByld(R.id.myTextView!1); 


if(getRequestedOrientation()==-1) 

{ 
mTlextView01.setText(getResources().getText 
(R.string.str_ err 1001)); 

} 


2) 定义 setOnClickListenermew Button.OnClickListener() 方 法 ， 当 用 户 单 击 按钮 后 开始 旋 
转 屏 幕 画 面 。 具 体 代码 如 下 。 
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mButton01.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 


public void onClick(View arg0) 


{ 
上 # 方法 一 : 重 写 getRequestedOrientation( ) 方 法 */ 


/hx 若 无 法 取得 screenOrientation 属性 */ 

if(getRequestedOrientation()==-1) 

{ 
旋 提示 无 法 进行 画面 旋转 功能 ， 因 无 法 判别 Orientation */ 
mfTlextView01.setText(getResources().getText 
(R.string.str_err 1001)); 


} 


else 
{ 
if(getRequestedOrientation()== 
ActivityInfo.SCREEN ORIENTATION LANDSCAPE) 
{ 
旋 若 当下 为 横 排 ， 则 更 改 为 竖 排 显现 */ 
setRequestedOrientation 
(ActivityInfo.SCREEN ORIENTATION PORTRAIT); 
} 
else if(getRequestedOrientation()== 
ActivityInfo.SCREEN ORIENTATION PORTRAIT) 
{ 
上/# 若 当 下 为 紧 排 ， 则 更 改 为 横 排 呈现 对 
setRequestedOrientation 
(ActivityInfo.SCREEN ORIENTATION LANDSCAPE); 
} 


人 
} 


3) 定义 setRequestedOrientation(int requestedOrientation) 方 法 ， 判 断 当前 要 更 改 的 屏幕 方 
辣 ， 实 现 旋 转 。 具 体 代码 如 下 。 


(QOverride 
public void setRequestedOrientation(int requestedOrientation) 


{ 


于 
让 


铸 判断 要 更 改 的 方向 ， 以 Toast 提示 */ 
switch(requestedOrientation) 


{ 
/# 更 改 为 LANDSCAPE (横向 显示 ) */ 
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case (ActivityInfo.SCREEN ORIENTATION LANDSCAPE): 
mMakeTextToast 
( 
getResources().getText(R.string.str msg1).toString(), 
false 
); 
break; 
人/# 更 改 为 PORTRAIT (直立 显示 ) */ 
case (ActivityInfo.SCREEN ORIENTATION PORTRAIT): 


mMakeTextToast 
( 
getResources().getText(R.string.str msg2).toString(), 
false 
); 
break; 
} 
super.setRequestedOrientation(requestedOrientation); 
} 


4) 定义 getRequestedOrientation(int requestedOrientatiom) 方 法 ， 获 取 当 前 屏幕 的 方向 。 
体 代 码 如 下 。 
(@Override 
public int getRequestedOrientation() 


{ 


/* 此 重 写 getRequestedOrientation(0) 方 法 ， 可 取得 当下 屏幕 的 方向 */ 
return super.getRequestedOrientation(); 


} 
public void mMakeTextToast(String str, boolean isLong) 
{ 
if(isLong==true) 
{ 
‘Toast.makeText(example18.this, str, Toast.LENGTH LONG).show(); 
} 
else 
{ 
Toast.makeText(example18.this, str, Toast.LENGTH SHORT).show(); 
} 
} 


} 


接 下 来 需要 在 文件 AndroidManifest.xml 中 设置 Activity 的 Android:screenOrientation 
性 ， 具 体 代码 如 下 。 


<activity 
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android:name=".examplel8" 
android:label="(@string/app_name" 
android:screenOrientation="portrait" 

> 


执行 后 会 显示 对 应 的 按钮 ， 如 图 5-42 所 示 。 单 击 “ 旋 转 处 理 ” 按 钮 后 会 实现 屏幕 的 旋 
转 ， 如 图 5-43 所 示 。 
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图 5-42 ”初始 效果 图 5-43 ”实现 旋转 


二 


在 手机 应 用 中 ， 为 了 特殊 需求 而 需要 获取 手机 和 网 络 的 相关 信息 。 在 本 节 的 内 容 中 ， 将 
通过 一 个 具体 实例 的 实现 ， 介 绍 获 取 网 络 和 手机 信息 的 基本 流程 。 本 实例 的 源 代 码 保存 在 
“光盘 :daimaNSvexample19”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


5.19.1 ”实现 原理 


在 本 实例 中 ， 将 使 用 TelephonyManager 类 来 获取 网 络 信息 。 另 外 ， 还 可 以 通过 Android 
API 提供 的 Android.provider Setting.System 类 来 获取 蓝牙 和 网 络 的 相关 信息 。 将 以 getSystem 
Service() 方 法 来 获取 TelephonyManage 对 象 ， 然 后 通过 TelephonyManage 的 方法 来 获取 和 电 
音 有 关 的 网 络 信息 。 然 后 通过 Android.provider Setting. System.getString0) 来 获取 手机 的 相关 设 


置信 息 ， 并 将 获取 的 信息 存 入 自 定 义 的 MyAdapter 中 ， 最 后 将 setListAdapter 〈 设 置 数据 源 方 
法 ) 内 的 信息 显示 在 ListView 中 。 


5.19.2 具体 实现 
本 实例 的 主 程序 文件 是 example19.java 和 MyAdapterjava， 下 面 开 始 讲解 其 具体 实现 
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1. 文件 example19.java 

下 面 先 讲解 文件 example19.java 的 具体 实现 流程 。 

1) 先 引 入 相关 类 ， 然 后 载 入 main.xml 布局 。 具 体 代 码 如 下 。 


public class example19 extends ListActivity 


{ 
private TelephonyManager tel Mgr; 


private List<String> item=new ArrayList<String>(); 
private List<String> value=new ArrayList<String>(); 


@Suppress Warnings("static-access") 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/# 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


telMgr = (TelephonyManager)getSystemService( TELEPHONY SERVICE); 


具体 包含 下 面 的 信息 。 


各 取得 的 信息 写 入 List (列表 ) 中 ， 
得 手机 电话 号 码 。 

得 电信 网 络 国 别 。 

得 电 信 公 司 代码 。 

导电 信 公 司 名 称 。 

和 通信 类 型 。 

得 手机 IMEI。 

得 手机 IMEI。 

得 IMEI SV。 


MD 
Ww 
GY 


SN 


SN 


SN 


证 
六 
iy 


得 Wi-Fi 状态 。 

J 模式 是 否 打开 。 
得 数据 漫游 是 否 打开 。 
上 述 信息 都 无 法 获得 ， 则 输出 “无 法 取得 ”的 提示 。 有 具体 代码 如 下 。 


此 将 取得 的 信息 写 入 List 中 */ 
item.add(getResources().getText(R.string.str list0).toString()); 
if(telIMegr.getLinel Number()!=null) 


{ 


~ 
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value.add(tel Mgr.getLinel Number()); 
} 


else 
{ 

value.add(" 无 法 取得 "); 
} 


[al 


席 取得 电信 网 络 国 别 * 
item.add(getResources().getText(R.string.str list1).toString()); 


if(tel Mgr.getNetworkCountrylso().equals("")) 


{ 
value.add(" 无 法 取得 "); 


} 


else 
{ 

value.add(""+tel Mgr.getNetworkCountryIso()); 
} 


席 取得 电信 公司 代码 * 
item.add(getResources().getText(R.string.str list2).toString()); 
if(telI Mgr.getNetworkOperator().equals("")) 
{ 
value.add(" 无 法 取得 "); 
} 
else 
{ 
value.add(tel Mgr.getNetworkOperator()); 


} 


席 取得 电信 公司 名 称 */ 
item.add(getResources().getText(R.string.str list3).toString()); 
if(tel Mgr.getNetworkOperatorName().equals("")) 
{ 
value.add(" 无 法 取得 "); 
} 
else 
{ 
value.add(tel Mgr.getNetworkOperatorName()); 


} 


席 取得 行动 通信 类 型 */ 
item.add(getResources().getText(R.string.str list4).toString()); 
if(telI Mgr.getPhoneType()==telI Mgr.PHONE TYPE GSM) 
{ 

value.add("GSM"); 


} 


看 国 229 


Android 开发 完全 实战 宝典 
else 
value.add(" 未 知 "); 
} 


启 取得 网 络 类 型 激 
item.add(getResources().getText(R.string.str list$).toString()); 
if(telI Mgr.getNetworkType()—telMgr.NETWORK TYPE EDGE) 
{ 
value.add("EDGE"); 
} 
else if(telI Mgr.getNetworkType()==telIMgr.NETWORK TYPE GPRS) 
{ 
value.add("GPRS'"); 
} 
else if(teI Mgr.getNetworkType()==telIMgr.NETWORK TYPE UMTS) 
{ 
value.add("UMTS"); 
} 
else if(telI Mgr.getNetworkType()==4) 
{ 
value.add("HSDPA"); 
} 
else 
{ 
value.add(" 未 知 "); 
} 


席 取得 漫游 状态 */ 
item.add(getResources().getText(R.string.str list6).toString()); 
if(telIMgr.isNetworkRoaming()) 
{ 
value.add(" 漫 游 中 "); 
} 
else{ 
value.add(" 无 漫游 "); 
} 


入 取得 手机 IMEI */ 
item.add(getResources().getText(R.string.str list7).toString()); 
value.add(tel Meger.getDeviceld()); 


族 取 得 IMEI SV */ 
item.add(getResources().getText(R.string.str list8).toString()); 
if(telI Megr.getDeviceSoftwareVersion()!=null) 

{ 


230 国 国 


第 .5 章 
value.add(telMgrgetDeviceSoftwareVersion()); 
} 


else 


{ 
value.add(" 无 法 取得 "); 


入 取得 手机 IMSI */ 
item.add(getResources().getText(R.string.str list9).toString()); 
if(telI Megr.getSubscriberId()!=null) 
{ 

value.add(tel Mgr.getSubscriberId()); 
} 


else 


{ 
value.add(" 无 法 取得 "); 


/* 取得 ContentResolver */ 
ContentResolver cv = examplel19.this.getContentResolver(); 


= 


String tmpS=""; 


雍 取得 蓝牙 状态 */ 
item.add(getResources().getText(R.string.str list10) 
toString()); 
tmpS=android.provider. Settings.System.getString(¢v, 
android.provider. Settings.System.BLUETOOTH. ON); 
if(tmpS.equals("1")) 
{ 
value.add(" 已 打开 "); 
} 
else{ 


value.add(" 未 打开 "); 


睛 取得 WIFI 状态 * 
item.add(getResources().getText(R.string.str list11) 
toString()); 
tmpS=android.provider. Settings.System.getString(¢v, 
android.provider.Settings.System.WIFI ON); 
if(tmpS.equals("1")) 
{ 
value.add(" 已 打开 "); 
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}》 


else{ 


value.add(" 未 打开 "); 


席 取得 飞行 模式 是 否 打开 */ 
item.add(getResources().getText(R.string.str list12) 
toString()); 
tmpS=android.provider.Settings.System.getString(¢v, 
android.provider.Settings.System.AIRPLANE MODE ON); 
if(tmpS.equals("1")) 


{ 

value.add(" 打 开 中 "); 
) 
else{ 

value.add(" 未 打开 "); 
} 


入 取得 数据 漫游 是 否 打 开 *%/ 
item.add(getResources().getText(R.string.str list13) 
toString()); 
tmpS=android.provider. Settings.System.getString(¢v, 
android.provider.Settings.System.DATA ROAMING); 
if(tmpS.equals("1")) 


{ 

value.add(" 打 开 中 "); 
} 
else{ 

value.add(" 未 打开 "); 
} 


3) 使 用 自 定义 的 MyAdapter 来 将 数据 传 入 ListActivity， 具 体 代码 如 下 。 


setListAdapter(new MyAdapter(this,item,value)); 
} 
} 


2. 文件 MyAdapterjava 
下 面 先 讲解 文件 MyAdapterjava 的 具体 实现 流程 。 


1) 先 加 载 相关 类 ， 然 后 自 定 义 Adapter， 并 继承 于 android.widget.BaseAdapter。] 
人 码 如 下 。 


/* 自 定 义 Adapter， 继 承 于 android.widget.BaseAdapter */ 
public class MyAdapter extends BaseAdapter 
{ 
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入 变量 声明 % 
private LayoutInflater mInflater; 


private List<String> items; 

private List<String> values; 

/* MyAdapter 的 构造 器 ， 传 入 三 个 参数 *%/ 

public MyAdapter(Context contexbList<String> item, 
List<String> value) 


入 参数 初始 化 忆 
mInfater=LayoutInflater.from(contexb); 
items = item; 

values = value; 


} 


2) 通过 @Override 分 别 覆 盖 方 法 getCount()、getItem(int position)、getItemId(int position) 
和 getView(int position,View convertView,ViewGroup par)。 上 其 体 代 码 如 下 。 


/# 因 继 承 BaseAdapter， 需 覆盖 以 下 方法 六 
(QOverride 
public int getCount() 


{ 


return items.size(); 


(QOverride 
public Object getItem(int position) 
{ 

return items.get(position); 


} 


(@Override 
public long getItemld(int position) 
{ 


return position; 


(QOverride 
public View getView(int position, View convertView,ViewGroup par) 


{ 
ViewHolder holder; 


if(convertView == null) 
{ 
/* 使 用 自 定义 的 fle row 作为 Layout */ 
convertView = mInflater.inflate(R.layout.row_layout,null); 
/* 初始 化 holder 的 text 与 icon */ 
holder = new ViewHolder(); 
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holdertext1=(TIextView)convertView.findViewById(R.id.myTextl); 
holdertext2=(TIextView)convertView.findViewById(R.id.myText2); 


convertView.setTag(holder); 
} 
else 
{ 
holder = (ViewHolder) convertView.getTag(); 
} 
入 设置 要 显示 的 信息 */ 
holder.textl1.setText(items.get(position).toString()); 
holder.text2.setText(values.get(position).toString()); 


return convertView; 


/* class ViewHolder */ 
private class ViewHolder 
{ 
/* text1: 信息 名 称 
* text2: 信息 内 容 */ 
TextView textl; 
TextView text2; 
} 
} 


执行 后 会 按照 指定 样式 显示 获取 的 网 络 信息 和 手机 信息 ， 如 图 5-44 所 示 。 


us 


310260 


Androld 


图 5-44 ”获取 的 信息 
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在 手机 应 用 中 ， 有 很 多 的 自动 服务 功能 。 例 如 ， 剩 余 电量 提示 、 存 储 卡 容量 提示 和 黑 名 
单 自 动 屏蔽 等 。 通 过 这 些 自动 服务 功能 ， 很 好 的 为 用 户 提供 了 人 性 化 的 服务 ， 整 个 操作 过 程 
更 加 方便 。 在 本 节 的 内 容 中 ， 将 通过 几 个 典型 实例 的 实现 ， 来 详细 介绍 这 些 自动 手机 服务 的 
实现 流程 。 
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在 当前 手机 应 用 中 ， 如 果 收 到 了 短信 ， 系 统 将 自动 在 手机 上 显示 “有 短信 ”的 提示 。 在 
本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 短信 提醒 的 基本 流程 。 本 实例 的 源 代 码 保 
存在 “光盘 :\daima\6\examplel1”， 下面 开始 讲解 本 实例 的 具体 实现 流程 。 


6.1.1 实现 原理 


在 手机 系统 中 ， 存 在 广播 系统 BroadcastReceiver 〈 广 播 接 收 者 )， 它 实时 监听 手机 内 的 
短信 状况 。 当 手机 收 到 短信 后 ， 会 通过 Notification 在 状态 栏 中 显示 短信 的 摘要 信息 。 

在 本 实例 中 ， 会 将 接收 到 的 短信 对 象 解析 为 可 以 识别 发 信人 号 码 和 短信 正文 的 字符 串 。 
本 实例 的 难点 是 如 何 向 系统 注册 一 个 常 驻 的 BroadcastReceiver 对 象 ， 然 后 在 后 台中 监听 短信 
事件 ， 最 后 将 短信 内 容 编译 出 来 。 


6.1.2 具体 实现 


本 实例 的 主 程序 文件 是 examplel.java 和 examplel SMSreceiverjava， 下 面 开始 讲解 其 具 
体 实 现代 码 。 

1. 文件 example1.java 

下 面 先 讲解 文件 examplel.java， 其 功能 是 在 文本 框 中 显示 “正在 等 待 接 收 短信 ...” 的 提 
示 。 有 具体 代码 如 下 。 


public class examplel extends Activity 


{ 
private TextView mTextView!; 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 
/# 通 过 findViewById 构造 器 创建 TextView 对 象 */ 
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ImTextViewl = (TextView) findViewByld(R.id.myTextView!1); 
mTextView1l.setText(" 现 在 正在 等 待 接收 信息 ..."); 
} 
} 


2. 文件 example1_SMSreceiver.java 

接 下 来 讲解 文件 examplel SMSreceiverjava， 下 面 开始 讲解 其 具体 实现 流程 。 

1) 先 引 用 BroadcastReceiver 类 ， 然 后 引用 telephoney.gsm.SmsMessage 对 象 来 收取 短 
言 ， 接 着 引用 Toast 类 来 告知 用 户 收 到 短信 。 有 具体 代码 如 下 。 


package irdc.examplel; 


/* 必 须 引 用 BroadcastReceiver 类 */ 

import android.content.BroadcastReceiver; 

import android.content.Context; 

import android.content. Intent; 

import android.os.Bundle; 

放 必 须 引 用 telephoney.gsm.SmsMessage 来 收取 短信 */ 
import android.telephony.gsm.SmsMessage; 

/必须 引用 Toast 类 来 告知 用 户 收 到 短信 */ 

import android.widget.Toast; 


2) 自 定 义 继 承 自 BroadcastReceiver 的 类 ， 用 于 监听 系统 服务 广播 的 信息 。 具 体 实现 流 
程 如 下 。 
第 一 步 : 声明 静态 字符 串 ， 并 使 用 android.provider.Telephony.SMS_RECEIVED 作为 
Action 为 短信 的 依据 。 

第 二 步 : 通过 站 语句 判断 传 来 的 Intent 是 否 为 短信 ， 如 果 是 ， 则 先 建 构 一 字符 串 集合 变 
量 sb， 然 后 接收 由 Intent 传 来 的 数据 。 

第 三 步 : 通过 于 语句 判断 Intent 是 有 数据 。 


具体 代码 如 下 。 


public class examplel SMSreceiver extends BroadcastReceiver 
{ 
放声 明 静 态 字 符 串 ， 并 使 用 android.provider.Telephony.SMS RECEIVED 
作为 Action 为 短信 的 依据 */ 
private static final String mACTION = 
"android.provider.Telephony.SMS RECEIVED"; 


二 


四 


(QOverride 
public void onReceive(Context context, Intent intent) 
{ 

/x* 判断 传 来 Intent 是 否 为 短信 */ 

if (intent.getAction().equals(mACTION)) 

{ 


/* 建 构 一 字符 串 集合 变量 sb*/ 
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StringBuilder sb = new StringBuilder(); 
访 接 收 由 Intent 传 来 的 数据 */ 
Bundle bundle = intent.getExtras(); 
访 判 断 Intent 是 有 数据 */ 
if (bundle != null) 
{ 
/*pdus 为 android 内 置 短信 参数 identifier 
* 通过 bundle.get(") 返 回 一 包含 pdus 的 对 象 */ 
Object[] myOBJpdus = (Object[]) bundle.get("pdus"); 
必 构 建 短信 对 象 array， 并 依据 收 到 的 对 象 长 度 来 创建 array 的 大 小 */ 
SmsMessage[] messages = new SmsMessage[myOBJpdus.length]; 
for (int i= 0; i<myOBJpdus.length; i++) 
{ 
messages[i| = 
SmsMessage.createFromPdu((byte[]) myOBJpdus[i]); 
} 


/* 将 送 来 的 短信 合并 ， 自 定义 信息 于 StringBuilder 当中 */ 
for (SmsMessage currentMessage : messages) 


{ 


sb.append(" 正 在 接收 到 来 自 :n"); 

族 来 电 者 的 电话 号 码 * 
sb.append(currentMessage.getDisplayOriginatingA ddress()); 
sb.append("n----- 发 米 的 短信 -----\n"); 

上 取得 传 来 信息 的 BODY */ 
sb.append(currentMessage.getDisplayMessageBody!()); 


} 
3) 以 _Notification(Toase) 显 示 短信 信息 ， 然 后 返回 主 Activity 界面 ， 并 使 其 以 一 个 全 新 
的 task《〈 进 程 ) 来 运行 。 有 具体 代码 如 下 。 


Toast.makeText 


( 


context, sb.toString(), Toast.LENGTH LONG 
).show(); 


/六 返回 主 Activity */ 
Intent 1= new Intent(context, examplel .class); 
2 使 其 以 一 个 全 新 的 进程 来 运行 */ 
1.addFlags(Intent.FLAG ACTIVITY NEW TASRK); 
context.startActivity(2); 
} 
} 
) 


接 下 来 还 需要 在 文件 AndroidManifest.xml 中 向 系统 注册 常 驻 的 receiver， 并 设置 这 个 
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receiver 的 intent-filter 名 为 “android.provider.Telephony.SMS_RECEIVED”。 男 外 ， 还 需要 设 
置 permission.RECEIVE_SMS 权限 。 具 体 代 码 如 下 。 

<!-- 建立 receiver 来 监听 系统 广播 信息 --> 

<receiver android:name="examplel SMSreceiver"> 


<! 一 设 定 要 捕捉 的 讯息 名 称 为 provider 中 Telephony.SMS_ RECEIVED --> 
<intent-filter> 


<action 
android:name="android.provider.Telephony.SMS RECEIVED"/> 
</intent-filter> 
</receiver> 
</application> 


<uses-permission android:name="android.permission.RECEIVE SMS"></uses-permission> 


执行 后 的 初始 效果 如 图 6-1 所 示 。 当 接收 到 短信 后 会 在 屏幕 中 显示 对 应 的 提示 信息 ， 如 
6-2 所 示 。 


example1 


正在 接收 到 来 自 : 


动 关 全 2:19Av 


图 6-1 初始 效果 图 6-2 短信 提示 
同时 在 系统 的 短信 栏目 中 会 显示 收 到 的 短信 ， 如 图 6-3 所 示 。 


注意 : 在 具体 测试 时 ， 需 要 同时 运行 两 个 模拟 器 ， 一 个 用 于 发 送 短信 ， 另 外 一 个 接收 。 
但 是 如 果 机 器 太 慢 ， 无 法 尼 动 两 个 模拟 器 。 也 可 以 只 局 动 一 个 模拟 器 。 然 后 在 Eclipse 菜单 
中 依次 单 击 “windows” 一 “Show view” 一 “other” 一 “Android” 一 “Emulator Control”， 
打开 “Emulator Control” 面 板 。 在 Telephony Actions 分 组 框 中 ，Voice 是 呼叫 ，SMS 是 发 送 
短信 。Incoming number 是 模拟 器 的 端口 号 ， 也 可 以 使 用 这 个 功能 给 的 模拟 器 拨打 电话 或 发 
送 短信 。 如 图 6-4 所 示 . 
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在 选 购 手机 时 ， 电 池 容 量 是 一 个 十 分 重要 的 因素 。 在 使 用 过 程 中 ， 最 害怕 手机 没 电 而 影 
响 业 务 。 为 此 ， 显 示 电 池 容 量 的 应 用 变 得 愈 发 重要 了 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实 
例 的 实现 ， 介 绍 实现 查看 手机 电池 容量 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :daiman6' 
example2”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


6.2.1 ”实现 原理 


Android API 里 面 的 BroadcastReceiver 类 和 Button 的 Listener 类 似 ， 当 Receiver 被 注册 
后 会 在 后 台 等 待 被 其 他 程序 调用 。 当 指定 要 捕捉 的 Action 发 生 时 ，Receiver 就 会 被 调用 ， 并 
运行 onReseiver 来 实现 内 部 的 程序 。 
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在 本 实例 中 ， 将 利用 BroadcastReceiver 的 特性 来 获取 手机 电池 的 容量 。 通 过 注册 
BroadcastReceiver 时 设置 的 IntentFilter 来 获取 系统 发 出 的 IntentACTION BATTERY 
CHANGED， 然 后 获取 电池 的 容量 


6.2.2 具体 实现 
本 实例 的 主 程序 文件 是 example2.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 分 别 声明 3 个 变量 intLevel、intScale 和 mButton01， 然 后 创建 BroadcastReceiver 对 
象 ， 如 果 捕 捉 到 的 Action 是 ACTION BATTERY CHANGED， 则 运行 onBatteryInfo 
Receiver()。 具 体 代 码 如 下 


public class example2 extends Activity 


{ 
* 变量 声明 */ 


private int intLevel; 


; 


private int intScale; 
private Button mButton01; 


/* 创建 BroadcastReceiver */ 
private BroadcastReceiver mBatInfoReceiver=new BroadcastReceiver() 


{ 


public void onReceive(Context context, Intent intent) 
{ 
String action = intent.getAction(); 
/* 如 果 捕 捉 到 的 Action 是 ACTION BATTERY_CHANGED， 
* 就 运行 onBatteryInfoReceiver0 方 法 */ 
if (Intent.ACTION BATTERY CHANGED.equals(action)) 
{ 


intLevel = intent.getIntExtra("level", 0); 


intScale = intent.getIntExtra("scale", 100); 
onBatteryInfoReceiver(intLevel,intScale); 
} 
} 
上 


2) 在 onCreate0 中 载 入 主 布局 文件 main.xml， 然 后 初始 化 Button 和 设置 单 击 后 的 动 
作 ， 并 注册 一 个 系统 BroadcastReceiver， 用 于 访问 电池 容量 。 有 具体 代码 如 下 。 


可 
O 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
族 载 入 布局 文件 main.xml */ 


setContentView(R.layout.main); 
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久 初始 化 Button， 并 设置 单 击 后 的 动作 光 
mButton01 = (Button)findViewById(R.id.myButton!); 
mButton01.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
public void onClick(View V) 
{ 
# 注册 一 个 系统 BroadcastReceiver， 作 为 访问 电池 容量 之 用 */ 
registerReceiver 
( 
mBatInfoReceiver, 
new IntentFilter(Intent.ACTION BATTERY CHANGED) 
); 
} 
); 


} 


3) 定义 方法 onBatteryInfoReceiver()， 当 捕捉 到 ACTION_BATTERY_CHANGED 时 要 运 
行 这 个 方法 ， 首 先 创建 一 个 背景 模糊 的 Window 窗口 ， 且 将 对 话 窗口 放 在 前 景 ， 然 后 将 取得 
的 电池 计量 显示 于 Dialog 对 话 框 中 ， 最 后 设置 返回 主 画 面 的 按钮 。 具 体 代码 如 下 。 

/* 捕捉 到 ACTION BATTERY _ CHANGED 时 要 运行 的 方法 闷 


public void onBatteryInfoReceiver(int intLevel, int intScale) 


{ 


席 创建 跳出 的 对 话 窗 口 *%/ 

final Dialog d= new Dialog(example2.this); 
d.setTitle(R.string.str_ dialog title); 
d.setContentView(R.layout.mydialog); 


旋 创建 一 个 背景 模糊 的 Window 窗口 ， 且 将 对 话 窗 口 放 在 前 景 */ 
Window window = d.getWindow(); 


window.setFlags 

( 
WindowManager.LayoutParams.FLAG BLUR BEHIND, 
WindowManager.LayoutParams.FLAG BLUR BEHIND 


由 


翌 将 取得 的 电池 容量 显示 于 对 话 框 中 *%/ 
TextView mTextView02=(TextView)d.findViewById(R.id.myTextView2); 
mTextView02.setText 
( 
getResources().getText(R.string.str dialog body)+ 
String.valueOf(intLevel * 100/ intScale) + "%" 


由 


和 的 按钮 */ 


国 


和 # 设置 返回 主 
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Button mButton02 = (Button)d.findViewById(R.id.myButton2); 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 


(QOverride 
public void onClick(View v) 


上 
/* 反 注 册 Receiver， 并 关闭 对 话 窗口 */ 
unregisterReceiver(mBatInfoReceiver); 
d.dismiss(); 
} 
); 
d.show(); 
】 
} 


执行 后 的 初始 效果 如 
6-6 所 示 。 


图 6-5 所 示 。 当 单 击 “ 获 取 ” 按 钮 后 会 显示 当前 电池 的 容量 ， 如 


| 肥大 提 228w 


六 交 @ 228m 


example2 


获取 电池 剩余 容量 ? 


图 6-5 初始 效果 
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在 当前 手机 应 用 中 ， 短 信和 群发 是 一 个 十 分 重要 的 功能 。 在 本 节 的 内 容 9 
体 实例 的 实现 ， 介 绍 短信 群发 功能 的 基本 实现 流程 。 本 实例 的 源 代 人 码 保存 在 “光盘 :\daima\ 


6\example3”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


PF， 将 通过 一 个 具 


6.3.1 ”实现 原理 
在 本 实例 中 ， 当 单 击 “发 送 ” 按 钮 后 ， 会 


先 获取 手机 通讯 录 的 信息 ， 让 用 户 选择 短信 
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接收 者 。 选 好 后 返回 主 程序 ， 然 后 实现 短信 群发 功能 。 
在 使 用 本 实例 前 ， 首 先 在 通讯 录 添 加 一 些 联系 人 信息 ， 这 些 联系 人 作为 接收 短信 者 。 当 
用 户 选择 接收 者 后 ， 会 将 短信 发 送 到 目标 者 。 


6.3.2 具体 实现 


也 体 流 程 如 下 。 
1) 先 定 义 $ 个 变量 mTextView01、mTextView3、mTextView5、mButton01 和 mButton 
01， 然 后 分 别 为 上 述 变量 赋值 。 具 体 代 码 如 下 。 


public class example3 extends Activity 


{ 


private TextView mTextView01; 


private TextView mTextView3; 
private TextView mTextViews; 
private Button mButton01; 
private Button mButton02; 

人 # 先 声明 strMessage 为 String*/ 
String StrMessage; 


private static final int PICK CONTACT SUBACTIVITY = 2; 


(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


ImTextView01 = (TextView)findViewByld(R.id.myTextView!); 
mButton01 = (Button)findViewById(R.id.myButton!1); 
ImTextView3 = (TextView)findViewById(R.id.myTextView3); 
mButton02 = (Button)findViewById(R.id.myButton2); 
mlextViewS= (TextView)findViewById(R.id.myTextView5); 


2) 分 别 设置 2 个 Button 的 处 理事 件 ， 其 中 mButton01 用 于 获取 mTextView3 里 的 内 


容 ，mButton02 用 于 获取 mTextView5 里 的 内 容 。 具 体 代 码 如 下 。 


族 设 置 第 一 个 按钮 的 单 击 事件 */ 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 

Override 

public void onClick(View V) 

{ 


Uri uri = Uri.parse("content://contacts/people"); 

Intent intent = new Intent(Intent.ACTION PICK, uri); 
人 # 获 取 mTextView3 里 的 内 容 */ 

strMessage = mTextView3.getText().toString(); 
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startActivityForResult(intent, PICK CONTACT SUBACTIVITY); 
} 
»); 


证 设 置 第 二 个 按钮 的 单 击 事件 */ 
mButton02.setOnClickListener(new Button.OnClickListener() 


{ 


(@Override 
public void onClick(View v) 
{ 
Uri uri = Uri.parse("content://contacts/people"); 
Intent intent = new Intent(Intent.ACTION PICK, ur); 
/# 获 取 mTextViews 里 的 内 容 */ 
strMessage = mTextView5.getText().toString(); 


startActivityForResult(intent, PICK CONTACT SUBACTIVITY); 


} 


3) 在 获取 android.permission.READ_CONTACTS 的 权限 下 ， 通 过 try 语句 来 获取 通讯 录 
中 姓名 和 电话 ， 然 后 设置 要 寄 给 通讯 录 中 的 哪 一 个 号 码 ， 接 着 用 smsManager SendText 
Message() 方 法 发 送 短信 ， 最 后 用 Toast 来 显示 正在 传送 的 提示 。 具 体 代 码 如 下 。 


(@Override 
protected void onActivityResult 


(int requestCode, int resultCode, Intent data) 
{ 
switch (requestCode) 
{ 
case PICK CONTACT SUBACTIVITY: 
final Uri uriRet = data.getData(); 
if(uriRet != null) 
{ 
try 
{ 
/* 必须 要 有 android.permission.READ CONTACTS 权限 */ 
Cursor ¢ = managedQuery(uriRet, null, null, null, null); 


c.moveToFirst(); 

上 获取 通讯 录 的 姓名 *% 
String strName = 
c.getString(c.getColumnIndexOrThrow(People.NAME)); 
/* 获 取 通 讯 录 的 电话 党 

String strPhone = 
c.getString(c.getColumnIndexOrThrow(People.NUMBER)); 
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放 设 置 要 寄 给 通讯 录 里 的 电话 *%/ 
String strDestAddress = strPhone; 
System.out.println(strMessage); 


SmsManager smsManager = SmsManager.getDefault(); 


PendingIntent mPI = PendingIntent.getBroadcast 
(example3.this, 0, new Intent(), 0); 
旋 发 出 短信 */ 
smsManager.sendTextMessage 
( 
strDestAddress, null, strMessage, mPI, null 
); 
/# 用 Toast 显示 传送 中 */ 
Toast.makeText 


( 


example3 .this, 


getString(R.string.str msg)+strName, 
Toast.LENGTH SHORT 
).show(); 


mTextView01.setText(strNamet+":"+strPhone); 


catch(Exception e) 


mTextView01.setText(e.toString()); 


e.printStackTrace(); 
} 
break; 
} 
super.onActivityResult(requestCode, resultCode, data); 
} 
} 


手机 自动 服务 


接 下 来 还 需要 在 文件 AndroidManifest.xml 中 设置 允许 READ CONTACTS 权限 和 


SEND _SMS 权限 。 


L 体 代码 如 下 。 


<uses-permission android:name="android.permission.READ CONTACTS"></uses-permission> 


<uses-permission android:name="android.permission.SEND SMS"></uses-permission> 


执行 后 的 初始 效果 如 图 6-7 所 示 。 当 单 击 “发 送 ” 按 钮 后 会 显示 联系 人 界面 ， 如 图 6-8 


所 示 。 


在 上 述 实例 


系统 允许 用 户 创 建 若干 个 群 组 。 可 以 把 你 的 联系 人 们 很 轻松 的 放 入 各 种 不 同 的 群 组 里 面 。 


单 击 其 中 的 一 个 联系 人 后 ， 会 将 短信 发 送 给 他 ， 并 输出 发 送 提示 ， 如 图 6-9 所 示 。 
， 虽 然 实 现 了 短信 发 送 功能 ， 但 是 还 不 能 算是 群 组 发 送 。 实 际 上 Android 


Android 系统 之 所 以 提供 了 这 样 的 一 个 特性 ， 是 为 了 让 用 户 给 整 组 联系 人 群发 邮件 或 者 短 
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言 。 如 图 6-10 所 示 。 


Contacts 


管 好 人 


图 6-7 初始 效果 图 6-8 联系 人 界 盏 


由 Add new group 


9 
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Edit group 


Delete gro 


最 真心 的 祝福 和 与 !! 


Send group messages 


Send group mail 


图 6-9 已 发 送 图 6-10 Android 的 群 组 


当然 也 可 以 用 编程 的 方式 实现 群发 短信 。 我 们 可 以 用 字符 串 数 组 的 形式 保存 联系 人 数 
据 ， 也 可 以 通过 cursor 为 对 象 ， 然 后 通过 循环 的 方式 ， 在 取得 了 联系 人 数据 的 同时 就 传 出 指 
定 的 短信 内 容 。 
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短信 和 邮件 是 两 个 十 分 重要 的 应 用 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 
绍 通 过 短信 来 发 送 Email 通知 的 基本 实现 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\6\ 
example4”， 下面 开始 讲解 本 实例 的 具体 实现 流程 。 


6.4.1 实现 原理 

在 本 实例 中 ， 当 收 到 一 条 短信 后 ， 先 用 Toast 来 提示 获取 了 短信 ， 然 后 再 通过 Email 
发 送 到 用 户 的 邮箱 中 ， 这 样 就 可 以 将 重要 的 短信 在 邮箱 中 保存 ， 而 不 用 担心 手机 容量 的 问 
题 了 。 
在 具体 实现 上 ， 先 在 后 台 设 计 一 个 BroadcastReceiver 类 ， 用 于 等 待 接收 短信 ， 当 收 到 短 
信 后 ， 以 Bundle〈 捆 绑 ) 的 方式 来 封装 短信 的 内 容 ， 然 后 通过 Intent 方式 返回 给 主 程序 


sna 
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Activity。 因 为 Receiver 无 法 直接 发 送 Email， 所 以 需要 将 控制 权 给 主 程序 ， 通 过 主 程序 来 运 
行 发 送 Email。 当 主 程序 收 到 Bundle 后 ， 会 以 Bundle.getStringO 的 方法 来 取得 返回 短信 的 内 
容 ， 然 后 以 _Intent.setType(“‘plain/text”) 来 设置 要 打开 的 Intent 类 型 ， 并 以 关键 程序 Intent.put 
Extra(android.content.Intent.EXTRA _ EMAIL,strEmailReciver) 来 指定 要 打开 的 是 Email 所 需要 
的 Extra 参数 ， 当 Android 系统 收 到 这 些 参数 后 ， 就 会 打开 内 置 的 Email 发 送 程序 。 读 者 在 
此 需要 注意 的 是 ， 在 模拟 器 运行 后 会 显示 “No application can perform this action” 的 提示 ， 
而 在 真实 机 器 上 不 会 出 现 此 问题 。 


6.4.2 具体 实现 


本 实例 的 主 程序 文件 是 example4.java 和 example4SMSreceiver.java， 下 面 开始 讲解 其 具 
体 实 现代 码 。 

1. 文件 example1.java 

下 面 先 讲解 文件 examplel.java， 其 具体 实现 流程 如 下 。 

1) 声明 一 个 TextView，String 数组 与 两 个 文本 字符 串 变 量 ， 具 体 代 码 如 下 。 


public class example4 extends Activity 


{ 
/声明 一 个 TextView，String 数组 与 两 个 文本 字符 串 变 量 */ 
private TextView mTextView!; 


/ 琵 


public String[] strEmailReciver; 
public String strEmailSubject; 
public String strEmailBody; 


兰 


2) 在 onCreate() 方 法 中 通过 findViewByld 构造 器 来 创建 TextView 对 象 ， 并 通 
TextView 来 显示 “等 待 接收 短信 .…” 的 提示 。 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 通 过 findViewByld 构造 器 创建 TextView 对 象 */ 
ImTextViewl = (TextView) fndViewById(R.idmyTextViewl); 


1 


mTextView1.setText(" 等 待 接收 短信 ..."); 


3) 通过 try 语句 取得 短信 传 来 的 bundle， 并 取出 bundle 内 的 字符 串 ， 然 后 自 定 义 Intent 
来 运行 发 送 E-mail 的 工作 ， 同 时 设置 邮件 格式 为 “plain/text”， 并 分 别 取 得 EditText01,02， 
03,04 的 值 作为 收 件 人 地 址 、 附 件 、 主 题 和 正文 ， 最 后 将 取得 的 字符 串 放 入 mEmailIntent 
中 。 具 体 代 码 如 下 。 


try 
{ 
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/# 取 得 短信 传 来 的 bundlew/ 

Bundle bundle = this.getIntent().getExtras(); 

if (bundle!= null) 

{ 
/# 将 bundle 内 的 字符 串 取 出 
String sb = bundle.getString("STR_ INPUT"); 
族 自 定义 一 Intent 来 运行 寄 送 E-mail 的 工作 */ 
Intent mEmailIntent = 
new Intent(android.content.Intent.ACTION SEND); 
放 设 置 邮件 格式 为 “plain/text”*/ 
mEmailIntent.setType("plain/text"); 


x 


/* 

* 取得 EditText01,02,03,04 的 值 作为 

* 收 件 和 人 地址、 附件 、 主 题 、 正 文 

3/ 

strEmailReciver =new String[]{"jay.mingchiehQ@gmail.com"}; 
strEmailSubject = "你 有 一 封 短信 !1"; 


strEmailBody = sb.toString(); 


TT 


4/ 


/# 将 取得 的 字符 串 放 入 mEmailIntent 中 */ 
mEmailIntent.putExtra(android.content.Intent.EXTRA EMAIL, 
strEmailReciver); 
mEmailIntent.putExtra(android.content.Intent.EXTRA SUBJECT, 
strEmailSubject); 
mEmailIntent.putExtra(android.content.Intent.EXTRA TEXT, 
strEmailBody); 

startActivity(Intent.createChooser(mEmailIntent, 
getResources().getString(R.string.str message))); 


} 


else 


{ 
finish(); 


} 


catch(Exception e) 


{ 


e.printStack Trace(); 


} 


} 


2. 文件 example4SMSreceiver.java 

接 下 来 讲解 文件 example4SMSreceiverjava， 其 具体 实现 流程 如 下 。 

1) 先 引 用 BroadcastReceiver 类 ， 然 后 引用 telephoney.gsm.SmsMessage 来 收取 短信 ， 引 
用 Toast 类 来 告知 用 户 收 到 短信 ， 有 具体 代码 如 下 。 
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package irdc.example4; 


/# 引 用 BroadcastReceiver 类 */ 

import android.content.BroadcastReceiver; 

import android.content.Context; 

import android.content.Intent; 

import android.os.Bundle; 

/# 引 用 telephoney.gsm.SmsMessage 来 收取 短信 */ 
import android.telephony.gsm.SmsMessage; 

/# 引 用 Toast 类 来 告知 用 户 收 到 短信 */ 

import android.widget.Toast; 


于 监听 系统 服务 广播 的 信息 ， 然 后 声明 静态 


二 


2) 自 定 义 继承 自 BroadcastReceiver 类 ， 
字符 串 并 作为 Action 的 依据 ， 有 具体 代码 如 下 。 


public class example4SMSreceiver extends BroadcastReceiver 
{ 
* android.provider.Telephony.SMS RECEIVED 
private static final String mACTION = 
"android.provider.Telephony.SMS RECEIVED"; 


private String str receive=" 收 到 短信 !"; 


3) 定义 onReceive(Context context，Intent intent) 用 于 获取 短信 ， 先 通过 于 语句 判断 传 来 
Intent 是 否 为 短信 。 如 果 是 ， 则 建构 一 字符 串 集 合 变量 sb 并 接收 由 Intent 传 来 的 数据 。 具 体 
代码 如 下 


Override 

public void onReceive(Context context, Intent intent) 

{ 
/IODO Auto-generated method stub 
Toast.makeText(context, str_receive.toString(), 
Toast.LENGTH LONG).show(); 


记 判 断 传 来 Intent 是 否 为 短信 */ 
if (intent.getAction().equals(mACTION)) 
\ 


让 建 构 一 字符 串 集 合 变 量 sb*/ 
StringBuilder sb = new StringBuilder(); 
放 接 收 由 Intent 传 来 的 数据 */ 

Bundle bundle = intent.getExtras(); 


9 用 站 语句 判断 Intent 是 否 有 数据 ， 用 pdus 作为 android 内 置 短信 参数 identifier 〈 能 
只 别 身 份 )， 并 通过 bundle.get(") 返 回 一 包含 pdus 的 对 和 象 。 具 体 代码 如 下 。 


/* 判 断 Intent 是 有 数据 */ 
if (bundle != null) 
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{ 
Object[] myOBJpdus = (Object[]) bundle.get("pdus"); 


5) 构造 短信 对 象 aray， 并 依据 收 到 的 对 象 长 度 来 创建 array 的 大 小 。 其 体 代码 如 下 。 


SmsMessage[] messages =new SmsMessage[myOBJpdus.length]; 


for (inti= 0; i<myOBJpdus.length; i++) 
{ 

messages[i] = 

SmsMessage.createFromPdu((byte[]) myOBJpdus[i]); 
} 


6) 将 传递 来 的 短信 合并 ， 然 后 自 定 义 信息 于 StringBuilder 当 
电话 号 码 ， 传 来 信息 的 BODY。 有 具体 代码 如 下 。 


for (SmsMessage currentMessage : messages) 

{ 
sb.append(" 接 收 到 来 自 :\n"); 
上 收 信 人 的 电话 号 码 */ 
sb.append(currentMessage.getDisplayOriginatingAddress()); 
sb.append("\n------ 传 来 的 短信 ------\n"); 
旋 取得 传 来 信息 的 内 容 */ 
sb.append(currentMessage.getDisplayMessageBody(O); 
Toast.makeText 
( 

context, sb.toString(), Toast.LENGTH LONG 

).show(); 

} 

} 


7) 用 Notification(Toase) 显 示 来 讯 信 息 ， 具 体 代 码 如 下 。 


Toast.makeText 
( 

context, sb.toString(), Toast.LENGTH LONG 
).show(); 


。 即 分 别 获取 收 信人 的 


8) 返回 主 Activity， 然 后 自 定 义 一 Bundle， 用 putString0 方 法 将 短信 存 入 自 定义 的 


bundle 内 ， 最 后 设置 Intent 的 Flag 以 一 个 全 新 的 task〈 进 程 ) 来 运行 。 


具体 代码 如 下 。 


Intent 1= new Intent(context, example4.class); 

/* 自 定义 一 Bundle*/ 

Bundle mbundle = new Bundle(); 

/* 将 短信 信息 以 putString0 方 法 存 入 自 定义 的 bundle 内 */ 
mbundle.putString("STR INPUT", sb.toString()); 

/* 将 自 定义 bundle 写 入 Intent 中 对 

i.putExtras(mbundle); 

放 设 置 mtent 的 Flag 以 一 个 全 新 的 进程 来 运行 */ 
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i.addFlags(Intent.FLAG ACTIVITY NEW _ TASK); 


context.startActivity(i); 
} 
} 
} 


最 后 ， 还 需要 在 文件 AndroidManifest.xml 中 向 系统 注册 一 个 常 驻 的 BroadcastReceiver， 
并 设置 这 个 Receiver 的 intent-filter， 让 其 SMSreceiver 针对 收 到 短信 事件 做 出 反应 ， 并 添加 
android.permission.RECEIVE_SMS 权限 。 具 体 代码 如 下 。 


<receiver android:name="example4ASMSreceiver"> 


<!-- 设 定 要 捕捉 的 讯息 名 称 为 provider 中 Telephony.SMS RECEIVED --> 


<intent-filter> 


<action 


android:name="android.provider.Telephony.SMS RECEIVED"/> 


</intent-filter> 
</receiver> 


</application> 


<uses-permission android:name="android.permission.RECEIVE SMS"></uses-permission> 


执行 后 可 以 向 其 发 送 一 条 短信 ， 收 到 短信 后 
图 6-11 所 示 。 


‘example4 


显示 提示 信息 /EC 并 生成 邮件 提示 o 如 
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\ 全/ 
ee)a 
AMmO 


Ee 到 放 全 人 全 人 全 芭 - 


als lo le lelnl) kk 
lele lll 
Em pe pe 


smle | 


电 35 加 时 


的 源 代码 保存 在 “光盘 :\daima\6\example5”， 下 面 开 


图 6-11 运行 效果 


在 当前 手机 应 用 中 ， 如 果 有 来 电 ， 则 会 在 屏幕 中 显示 
节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 实现 来 电 屏 幕 提 醒 的 基本 实现 流程 。 
台 讲 ; 


解 本 实例 的 具体 实现 流程 。 


拨打 用 户 的 姓名 等 基本 信息 。 在 本 


本 实例 
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6.5.1 ”实现 原理 
在 Android 中 ， 可 以 通过 PhoneStateListener 〈 监 听 电 话 状 态 的 方法 ) 提供 的 方法 来 监听 
来 电 状 态 。 在 具体 实施 时 ， 需 要 创建 PhoneStateListener 对 象 ， 并 重 写 其 中 的 onCallState 
Changed() 方 法 ， 并 通过 传 入 的 “state” 来 判断 来 电 状态 。 

要 获取 来 电 状态 ， 需 要 用 户 具有 读 取 电话 状态 的 权限 ， 否 则 不 能 成 功 获取 状态 。 在 有 具体 
实施 上 ， 需 要 在 模拟 器 中 先 添加 一 个 联系 人 记录 ， 并 为 其 起 名 。 这 样 当 电话 进来 后 ， 会 在 屏 
幕 中 显示 他 的 名 字 。 如 果 是 非 通讯 录 的 来 电 ， 则 在 屏幕 中 显示 Unknown Caller。 


6.5.2 TelephonyManager 和 PhoneStateListener 

开发 应 用 程序 的 时 候 ， 和 希望 能 够 监听 电话 的 呼 入 ， 以 便 执 行 暂 停 音乐 播放 器 等 操作 ， 当 

电话 结束 之 后 ， 再 次 恢复 播放 。Android 平台 可 以 通过 TelephonyManager 和 PhoneState 

Listener 来 完成 此 任务 。 
TelephonyManager 作为 一 个 Service 接口 提供 给 用 户 查 询 电话 相关 的 内 容 ， 例 如 IMET、 

LineNumberl 等 。 通 过 下 面 的 代码 即 可 获得 TelephonyManager 的 实例 。 


TelephonyManager mTelephonyMgr = (TelephonyManager) this 
.getSystemService(Context.TELEPHONY SERVICE); 


-二 


j 的 监听 器 ， 用 来 监听 电话 的 状态 ， 如 


< 
AN 


在 Android 平台 中 ，PhoneStateListener 是 个 入 
呼叫 状态 和 连接 服务 等 。 其 方法 如 下 。 


public void onCallForwardingIndicatorChanged(boolean cfi) 
public void onCallStateChanged(int state, String incomingNumber) 
public void onCellLocationChanged(CellLocation location) 

public void onDataActivity(int direction) 


public void onDataConnectionStateChanged(int state) 

public void on MessageWaitingIndicatorChanged(boolean mwi) 
public void onServiceStateChanged(ServiceState serviceState) 
public void onSignalStrengthChanged(int asu) 


这 里 只 需要 履 盖 onCallStateChanged0 方 法 即 可 监听 呼叫 状态 。 在 TelephonyManager 中 
定义 了 三 种 状态 ， 分 别 是 振 铃 (RINGING)、 摘 机 COFFHOOK ) 和 空闲 〈IDLE)， 通 过 
“state” 的 值 就 知道 现在 的 电话 状态 了 。 

获得 了 TelephonyManager 接口 之 后 ， 调 用 listen0 方 法 即 可 监听 电话 状态 。 


mTelephonyMgr.listen(new TeleListener(), 
PhoneStateListener.LISTEN CALL STATE); 


6.5.3 ”具体 实现 

本 实例 的 主 程序 文件 是 example5.java， 下 面 开始 讲 解 其 具体 实现 流程 。 

1 ) 通过 方法 setContentView 来 引用 主 布局 文件 main.xml， 然 后 通过 myTextView1 显示 
提示 。 有 其 体 代码 如 下 。 
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public class exampleS extends Activity 


{ 


private TextView myTextView!; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


myTextViewl = (TextView) findViewById(R.id.myTextView!1); 


2) 定义 TelephonyManager 对 象 tm， 用 于 获取 电话 服务 ， 然 后 通过 tm.listen 来 注册 电话 
通信 Listener。 具 体 代 码 如 下 。 


诺 添加 自己 实现 的 PhoneStateListener */ 
exPhoneCallListener myPhoneCallListener = 


new exPhoneCallListener(); 


席 取得 电话 服务 */ 

TelephonyManager tm = 
(TelephonyManager) this.getSystemService 
(Context.TELEPHONY SERVICE); 


/# 注册 电话 通信 Listener */ 
tm.listen 


( 


myPhoneCallListener, 
PhoneStateListener.LISTEN CALL STATE 


;» 


} 


3) 使 用 内 部 类 来 继承 PhoneStateListener， 重 写 onCallStateChanged 方法 ， 这 样 当 状 态 改 


变 时 改变 myTextView1 的 文字 及 颜色 。 然 后 分 别 设置 无 任何 状态 、 接 起 电话 和 电话 呼 入 的 显 
示 。 有 具体 代码 如 下 。 


public class exPhoneCallListener extends PhoneStateListener 
{ 
/* 重 写 onCallStateChanged 
当 状 态 改 变 时 改变 myTextView1 的 文字 及 颜色 */ 
public void onCallStateChanged(int state, String incomingNumber) 
{ 
switch (state) 


{ 


国 国 253 


Android 开发 完全 实战 宝典 

庆 无 任何 状态 时 * 

case TelephonyManager.CALL STATE IDLE: 
myTextViewl.setTextColor 
( 

getResources().getColor(R.drawable.red) 

); 
myTextViewl.setText("CALL STATE IDLE"); 
break; 

庶 接 起 电话 时 */ 

case TelephonyManager.CALL STATE OFFHOOK: 
myTextViewl.setTextColor 
( 

getResources().getColor(R.drawable.greem) 

); 
myTextViewl.setText("CALL STATE OFFHOOK"); 
break; 

雍 电话 呼 入 时 */ 

case TelephonyManager.CALL STATE RINGING: 
getContactPeople(incomingNumben); 
break; 

default: 
break; 


} 


super.onCallStateChanged(state, incoming Number); 


} 


4) 通过 方法 getContactPeople(String incomingNumben) 来 获取 机 器 内 的 联系 人 信息 ， 然 后 
在 光标 里 存放 字段 名 称 。 有 具体 代码 如 下 。 


private void getContactPeople(String incomingNumber) 

{ 
myTextViewl.setTextColor(Color.BLUE); 
ContentResolver contentResolver = getContentResolver(); 


Cursor cursor = null; 


诺 cursor 里 要 放 的 字段 名 称 */ 
String[] projection = new String[] 
{ 
Contacts.People. ID, 
Contacts.People. NAME, 
Contacts.People. NUMBER 


儿 
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5) 通过 来 电 号 码 查找 对 应 的 联系 人 ， 查 找到 则 显示 姓名 ， 没 有 查找 到 则 只 显示 号 码 。 
体 代码 如 下 。 


族 用 来 电 电话 号 码 去 找 该 联系 人 */ 
cursor = contentResolver.query 
( 
Contacts.People.CONTENT _ URI, projection, 
Contacts.People.NUMBER + "=?", 
new String[] 
{ 
incomingNumber 
} > 
Contacts.People.DEFAULT SORT ORDER 
); 


Ln 


/* 找 不 到 联系 人 * 
if (cursor.getCount() == 0) 
1 


myTextViewl.setText("unknown Number:" + incomingNumber); 


} 


else if (cursor.getCount() > 0) 


{ 


cursor.moveToFirst(); 
/* 在 projection 这 个 数组 里 名 字 是 放 在 第 1 个 位 置 */ 
String name = cursor.getString(1); 


myTextViewl.setText(name + ":" + incomingNumber); 


} 


接 下 来 还 需要 在 文件 AndroidManifest.xml 中 获取 如 下 两 个 权限 。 

口 读 取 通讯 录 权 限 : android.permission.READ _CONTACTS。 

口 获取 电话 状态 权限 : android.permission.READ PHONE _STATE。 
L 体 代码 如 下 。 


<uses-permission android:name="android.permission.READ CONTACTS"></uses-permission> 
<uses-permission android:name="android.permission.READ PHONE STATE"></uses-permission> 


执行 后 的 效果 如 图 6-12 所 示 ， 当 打 来 电话 后 会 显示 来 电 的 基本 信息 ， 如 图 6-13 所 示 。 
[= 碟 判 全 3:07Aw 
examples | 


example5 
CALL_STATE_IDLE 


图 6-12 ”初始 效果 
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Incoming call 


图 6-13 来 电 后 界面 


66 UU000000 


在 手机 应 用 中 ， 需 要 及 时 了 解 存储 卡 的 容量 信息 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实 
例 的 实现 ， 介 绍 编程 实现 获取 存储 卡 容量 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\ 
6\example6”， 下 面 开 始 讲 解 本 实例 的 具体 实现 流程 。 
6.6.1 ”实现 原理 

存储 卡 是 可 以 随时 插 拔 的 ， 每 次 插 拔 时 会 对 操作 系统 进行 ACTION broadcast。 在 本 实例 
中 ， 将 通过 StatFs 文件 系统 的 方法 来 取得 MicroSD 存储 卡 的 剩余 容量 。 并 且 在 有 具体 实施 时 ， 
需要 首先 判断 是 否 安装 存储 卡 ， 如 果 不 存在 则 直接 不 计算 。 为 了 更 好 的 显示 容量 ， 在 布局 中 
插入 了 一 个 ProgressBar 进度 条 组 件 ， 这 样 将 一 目 了 然 。 
6.6.2 具体 实现 

本 实例 的 主 程序 文件 是 example6.java， 下 面 开 始 讲解 其 具体 实现 流程 。 

1) 先 分 别 定义 3 个 参数 myButton、myProgressBar 和 myTextView， 然 后 通过 findView 
Byld 构造 3 个 对 象 myButton、myProgressBar 和 myTextView。 具 体 代码 如 下 。 


public class example6 extends Activity 


{ 
private Button myButton; 
private ProgressBar myProgressBar; 
private TextView myTextView; 
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(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


myButton = (Button) fndViewById(R.id.myButton); 
myProgressBar = (ProgressBar) findViewById(R.id.myProgressBar); 
myTextView = (TextView) findViewById(R.id.myTextView); 


po 


2) 定义 setOnClickListener 事件 ， 用 于 触发 按钮 单 击 事 件 处 理 程序 。 具 体 代 码 如 下 。 


myButton.setOnClickListener(new Button.OnClickListener() 


{ 


Override 
public void onClick(View arg0) 
{ 


showSize(); 


); 


by 


3) 定义 方法 showSize0， 用 于 显示 存储 卡 的 容量 大 小 。 流 程 


第 一 步 : 将 TextView 及 ProgressBar 设置 为 空 值 及 0。 
第 二 步 : 获取 SD CARD 文件 路 径 。 
第 三 步 : 通过 StatFs 方法 来 查看 文件 系统 空间 使 用 状况 。 

第 四 步 : 分 别 获取 全 部 的 Block 数量 和 已 使 用 的 Block 数量 
第 五 步 : 通过 getMax0 方 法 获取 在 main.xml 里 进度 条 设置 的 最 大 值 。 
第 六 步 : 显示 出 容量 信息 。 
第 七 部 : 如 果 没 有 SD 卡 则 输出 “SD CARD 已 删除 ”的 提示 。 
要 体 代 人 码 如 下 所 示 : 


private void showSize() 


{ 
/# 将 TextView 及 ProgressBar 设置 为 空 值 及 0 */ 
myTextView.setText(""); 


如 下 。 


T 


al 


DOODOODOOCD 


myProgressBar.setProgress(0); 
/# 判断 存储 卡 是 否 插入 */ 
if (Environment.getExternalStorageState().equals(Environment.MEDIA MOUNTED)) 
{ 
谍 ”取得 SD 卡 文件 路 径 ， 一 般 是 /sdcard */ 
File path = Environment.getExternalStorageDirectory(); 


作 StatFs 查看 文件 系统 空间 使 用 状况 * 


画 国 257 


Android 开发 完全 实战 宝典 


258 国画 


StatFs statFs = new StatFs(path.getPath()); 

/* Block 的 size*/ 

long blockSize = statF's.getBlockSize(); 

/* 总 的 Block 数量 */ 

long totalBlocks = statFs.getBlockCount(); 

/# ”已 使 用 的 Block 数量 */ 

long availableBlocks = statFs.getAvailableBlocks(); 


String[] total = fileSize(totalBlocks * blockSize); 
String[] available = fileSize(availableBlocks * blockSize); 


/* getMax 取得 在 main.xml 里 ProgressBar 设置 的 最 大 值 */ 
int ss = Integer.parseInt(available[0]) * myProgressBar.getMax() 
/Integer.parseInt(total[0]); 


myProgressBar.setProgress(ss); 

String text= "总 共 "+total[0] + total[1] + "\n"; 
text += "可 用 "+ available[0] + available[ 1]; 
myTextView.setText(text); 


} else if (Environment.getExternalStorageState().equals( 


{ 


/到 


Environment.MEDIA REMOVED)) 


String text = "SD CARD 已 删除 "; 
myTextView.setText(text); 


司 为 字符 串 数组 [0] 为 大 小 []] 为 单位 KB 或 MB%/ 


private String[] fileSize(long Size) 


{ 


String str 


= 
? 


if (size >= 1024) 


{ 


} 


str= "KB"; 
size /= 1024; 
if (size >= 1024) 
{ 
str= "MB",; 
size /= 1024; 


DecimalFormat formatter = new DecimalFormat(); 


Ws 


3 个 数字 用 ,分 隅 如 : 1,000 */ 


formatter.setGroupingSize(3); 


String result[] = new String[2]; 
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result[0] = formatter. format(size); 
result[1] = str; 
return result; 
} 
} 


执行 后 的 效果 如 图 6-14 所 示 。 


[A A lil wd i, 
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图 6-14 执行 效果 


Android 模拟 器 能 够 让 我 们 使 用 FAT32 格式 的 磁盘 镜像 作为 SD 卡 的 模拟 ， 具 体 过 程 
如 下 。 
1) 进入 Android SDK 目录 下 的 tools 子 目 录 ， 运 行 : 


mksdcard -1 sdcard S12M /your path for img/sdcard.img 


这 样 就 创建 了 一 个 512M 的 SD 卡 镜像 文件 。 

在 使 用 mksdcard 命令 时 要 注意 如 下 6 点 。 

口 mksdcard 命令 可 以 使 用 三 种 尺寸 : 字 节 、 千 字 节 和 兆 字 节 。 如 果 只 使 用 数字 ， 表 示 

字 节 。 后 面 还 可 以 跟 K， 如 262144K， 也 表示 256M。 

口 mksdcard 建立 的 虚拟 文件 最 小 为 383M， 也 就 是 说 ， 模 拟 器 只 支持 大 于 8M 的 虚拟 

文件 。 

口 1 命令 行 参数 表示 虚拟 磁盘 的 卷 标 ， 可 以 没有 该 参数 。 

口 虚拟 文件 的 扩展 名 可 以 是 任意 的 ， 如 mycard.abc。 

口 mksdcard 命令 不 会 自动 建立 不 存在 的 目录 ， 因 此 ， 在 执行 上 面 命令 之 前 ， 要 先 在 当 

前 目录 中 建立 一 个 card 目录 。 

口 mksdcard 命令 是 按 实际 大 小 生成 的 sdcard 虚拟 文件 。 也 就 是 说 ， 生 成 256M 的 虚拟 文 
件 的 尺寸 就 是 256M， 如 果 生 成 较 大 的 虚拟 文件 ， 要 看 看 自己 的 硬盘 空间 够 不 够 。 

2) 运行 模拟 器 的 时 候 指定 路 径 〈 注 意 需 要 完整 路 径 )。 
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emulator -sdcard /your path for img/sdcard.img 


这 样 模拟 器 中 就 可 以 使 用 
那么 如 何 复制 本 机 文件 到 SD 卡 ， 或 者 管理 


za 


“/sdcard” 这 个 路 径 来 指向 模拟 的 SD 卡 了 。 


ESD 卡 上 的 内 容 呢 ? 有 如 下 2 种 方案 。 
第 一 种 : mount 是 Linux 下 的 一 个 命令 ， 它 可 以 将 Windows 分 区 作为 Linux 的 一 个 “ 文 
件 ” 挂 接 到 Linux 的 一 个 空 文件 夹 下 ， 从 而 将 Windows 的 分 


区 和 /mnt 这 个 目录 联系 起 来 ， 因 


此 我 们 只 要 访问 这 个 文件 夹 ， 就 相当 于 访问 该 分 区 了 。 


mount -0 loop sdcard.img android sdcard 


这 样 管理 这 个 目录 就 是 管理 sdcard 内 容 了 。 


第 二 种 : 在 Windows 可 


是 环境 下 可 以 月 


的 命令 (这 个 命令 在 linux 下 面 也 可 以 用 )。 


adb push local file sdcard/remote file 


在 执行 完 


emulator -avd avd1 -sdcard card/mycard.img 


有 mtools 来 做 管理 ， 也 可 以 


二 HH 
市 


] android SDK 自 


上 面 的 命令 后 ， 执 行 下面 的 命令 局 动 Android 模拟 器 。 


如 果 在 开发 环境 〈Eclipse) 中 ， 可 以 在 “Run Configuration” 对 话 框 中 设置 启动 参数 ， 


当然 ， 也 可 以 在 Preferences 对 话 框 中 设置 默认 启动 参数 。 这 样 在 新 建立 的 Android 工程 中 
自动 加 入 了 装载 sdcard 虚拟 文 


个 的 命令 行 参 数 。 


就 


如 果 读 者 使 用 OPhone 虚拟 机 ， 设 置 的 方法 完全 一 样 的。 在 虚拟 机 中 的 Setting 里 看 看 
sdcard， 是 否 找到 。 那 么 如 何 查 看 sdcard 虚拟 设备 中 的 内 容 呢 ?方法 很 多 ， 最 简单 的 就 是 使 
用 Android Eclipse 插件 带 的 DDMS 透视 图 。 

Ge 

在 前 面 的 实例 中 ， 介 绍 了 短信 、 邮 件 通知 的 实现 过 程 。 同 理 ， 也 可 以 编写 一 个 程序 ， 当 
来 电 时 用 邮件 实现 通知 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 来 电邮 件 通知 
的 基本 流程 。 本 实例 的 源 代 人 码 保 存在 “光盘 :daimaN6\example7”， 下 面 开 始 讲解 本 实例 的 具 
体 实现 流程 。 

6.7.1 ”实现 原理 
在 本 实例 中 ， 通 过 TelephoneManage 来 判断 来 电 状 态 ， 并 实现 来 电 通 知 。 程 序 通 过 


Email 来 通知 


来 电 记录 ， 本 实例 继承 了 前 面 的 实例 ， 并 再 次 对 PhoneCallListener 类 实现 电话 
事件 判断 ， 并 根据 来 电 状 态 发 送 Email。 和 前 面 的 实 


网 一 样 ， 在 模拟 器 运行 后 会 显示 “No 


application can perform this action” 的 提示 ， 而 在 真实 机 器 上 不 会 出 现 此 问题 。 


6.7.2 具体 实现 


本 实例 的 主 程序 文人 


体 实现 代码 。 
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是 example7.java 和 examplel SMSreceiverjava， 下 面 开 始 讲 
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1) 先 定 义 三 个 参数 mTextView1、mEditText01 和 strEmailSubject， 并 分 别 为 后 两 个 赋 
值 。 有 具体 代码 如 下 。 


public class example7 extends Activity 


{ 


private TextView mTextView!l; 
private String mEditText01 ="IRDC@gmail.com"; 
private String strEmailSubject = "You have phone!!"; 


2) 定义 TelephonyManager 类 的 对 象 teIMgr， 用 于 获取 TELEPHONY_SERVICE 系统 信 
县 3 ~ 体 代码 如 下 [eo] 


/** Called when the activity is first created. */ 
Override 
public void onCreate(Bundle savedInstanceState) 
{ 

super.onCreate(savedInstanceState); 

setContentView(R.layout.main); 


mPhoneCallListener phoneListener=new mPhoneCallListener(); 
/# 对 象 telMgr， 用 于 获取 TELEPHONY SERVICE 系统 */ 
TelephonyManager telMgr = (TelephonyManager)getSystemService 
(TELEPHONY _ SERVICE); 
telMgr.listen(phoneListener, mPhoneCallListener. 
LISTEN_ CALL STATE); 


ImTextViewl = (TextView)findViewById(R.id.myTextView!1); 
} 


3) 使 用 PhoneCallListener 来 监听 电话 状态 更 改 事件 ，onCallStateChanged(0 方 法 的 具体 
实现 流程 如 下 。 


第 一 步 : 获取 电话 待机 状态 。 
第 二 步 : 获取 电话 通话 状态 。 
第 三 步 : 获取 电话 来 电 状态 。 
第 四 步 : 显示 号 码 。 

第 五 步 : 有 电话 时 发 送 邮件 。 
第 六 步 : 设置 收 信人 邮箱 地 址 。 
第 七 步 : 设置 邮件 标题 。 

第 八 步 : 设置 邮件 内 容 

第 九 步 : 实现 发 信和 处理。 

上 上 述 功能 的 具体 代码 如 下 。 


public class mPhoneCallListener extends PhoneStateListener 


{ 
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@Override 
public void onCallStateChanged(int state, String incoming Number) 
{ 
switch(state) 
{ 
席 ”获取 电话 待机 状态 */ 
case TelephonyManager.CALL STATE IDLE: 
mTextViewl.setText(R.string.str CALL STATE IDLE); 
break; 
从 获取 电话 通话 状态 六 
case TelephonyManager.CALL STATE OFFHOOK: 
mTextViewl.setText(R.string.str CALL STATE OFFHOOR); 
break; 
人/* ”获取 电话 来 电 状 态 */ 
case TelephonyManager.CALL STATE RINGING: 
mTextViewl.setText 
( 
伴 显 示 号 码 凡 
getResources().getText(R.string.str CALL STATE RINGING)+ 


incomingNumber 


» 


放 有 电话 时 发 送 邮 件 */ 
Intent mEmailIntent = new Intent(android.content.Intent 
.ACTION SEND); 

mEmailIntent.setType("'plain/text"); 

虚设 置 收 信人 邮箱 地 址 * 

mEmailIntent.putExtra(android.content.Intent.EXTRA EMAIL, 

new String[] {mEditText01.toString()}); 

刻 设 置 邮件 标题 */ 

mEmailIntent.putExtra(android.content.Intent.EXTRA SUBJECT, 

strEmailSubject); 

放 设 置 邮件 内 容 */ 

mEmailIntent.putExtra(android.content.Intent.EXTRA TEXT, 

R.string.str EmailBodytincomingNumber); 

放 实 现 发 信和 处理 */ 

startActivity(Intent.createChooser(mEmailIntent, 
getResources().getString(R.string.str message))); 


break; 
default: 
break; 
} 


super.onCallStateChanged(state, incomingNumber); 
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执行 后 的 初始 效果 如 图 6-15 所 示 ， 当 拨打 电话 后 会 显示 来 电信 息 ， 如 图 6-16 所 示 。 


Incoming call 


图 6-15 初始 效果 图 6-16 来 电 后 界面 


当 挂机 后 会 发 送 邮件 ， 当 然 因 为 是 模拟 器 ， 所 以 显示 “No application can perform this 
action” 的 提示 。 如 网 6-17 所 示 。 


No applications can perform 
this action. 


图 6-17 发 邮件 


CS 二 


介绍 了 和 存储 卡 有 关 的 操作 。 在 本 节 的 内 容 中 ， 将 进一步 讲解 对 SD 
存储 卡 的 操作 方法 ， 个 具体 实例 的 实现 ， 介 绍 对 内 存 和 内 存 卡 中 文件 进行 操作 的 基本 
流程 。 “光盘 :daimaN6\example8”， 下 面 开 始 讲解 本 实例 的 具体 实现 
流程 。 
6.8.1 ”实现 原理 

移动 手机 的 存储 控件 分 为 内 存 控件 和 存储 卡 控 件 ， 在 本 实例 中 将 添加 两 个 按钮 ， 分 别 用 
于 添加 和 删除 内 存 或 存储 卡 内 的 文件 。 并 且 在 实例 中 使 用 了 3 个 Activity， 主 程序 的 是 Entry 


Activity， 另 外 2 个 分 别 用 于 处 理 内 存 卡 和 存储 卡 。 


当 用 户 选择 内 存 或 存储 卡 后 ， 将 以 列表 形式 显示 里 面 的 所 有 的 目录 和 文件 名 ， 并 在 
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Android 


单 ， 实 现 添加 文件 功能 。 当 单 击 “ 删 除 ” 按 钮 后 ， 可 以 删除 指定 的 文件 。 


6.8.2 


具体 实现 


> 


本 实例 的 主 程序 文件 是 example8.java、example8 1.java 和 example8 2.java， 


解 其 具体 实现 流程 。 


1 


1) 


文件 example8.java 


下 面 先 看 文件 example8.java 的 具体 实现 流程 。 


先 定义 4 个 参数 对 象 myButton1 、myButton2 、fileDir 和 sdcardDir， 


findViewById 构造 两 个 对 象 myButtonl 和 myButton2。 有 具体 代码 如 下 。 


2) 


public class example8 extends Activity 
{ 

private Button myButton 1 ; 

private Button myButton2; 

private File fileDir; 

private File sdcardDir; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


myButtonl = (Button) findViewById(R.id.myButton1l); 
myButton2 = (Button) findViewById(R.id.myButton2); 


主 菜 单 中 显示 “添加 ”或 “删除 ”按钮 。 单 击 “ 添 加 ”按钮 后 会 显示 一 个 添加 菜 


F 面 开始 i 


然后 通过 


通过 getFilesDir0 方 法 取得 SD 卡 的 目录 ， 设 置 当 SD Card 无 插入 时 ，myButton2 处 


于 不 可 月 


3) 
如 下 。 
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状态。 其 体 代码 如 下 。 


/* 取得 目前 File 目录 */ 
fileDir = this.getFilesDir(); 


上 # 取得 SD 卡 的 目录 党 
sdcardDir = Environment.getExternalStorageDirectory(); 


/# 当 SD Card 无 插入 时 ， 将 myButton2 设 成 不 可 用 */ 
if (Environment.getExternalStorageState().equals(Environment.MEDIA REMOVED)) 


{ 
myButton2.setEnabled(false); 


} 
分 别 定 义 按 钮 单 击 处 理 


出品 


逢 件 setOnClickListener 和 setOnClickListener， 


具体 代码 
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myButtonl.setOnClickListener(new Button.OnClickListener() 


{ 


Override 
public void onClick(View arg0) 
{ 
String path = fileDir.getParent() + java.io.File.separator 
+ fileDir.getName(); 
showListActivity(path); 
) 
); 
myButton2.setOnClickListener(new Button.OnClickListener() 


{ 


(QOverride 

public void onClick(View arg0) 

{ 
String path = sdcardDir.getParent() + sdcardDir.getName(); 
showListActivity(path); 


); 
} 


4) 定义 方法 showListActivity(String path)， 定 义 一 个 Intent 类 对 象 intent， 然 后 将 路 径 传 


到 example_ 1。 具体 代码 如 下 。 


private void showListActivity(String path) 
{ 


Intent intent = new Intent(); 


intent.setClass(example8.this, example8 1.class); 
Bundle bundle = new Bundle(); 

上 将 路 径 传 到 example 1 */ 
bundle.putString("path", path); 


intent.putExtras(bundle); 
startActivity(intent); 


} 


2. 文件 example8_1.java 
接 下 来 讲解 文件 example8_1.java， 其 具体 实现 流程 如 下 。 


1) 主 Activity 传 来 的 path 字符 串 作 为 传 入 路 径 ， 如 果 路 径 不 存在 ， 则 使 朋 


来 创建 。 具 体 代码 如 下 。 
public class example8 1 extends ListActivity 


{ 
private List<String> items = null; 
private String path; 


日 java.io.File 
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protected final static int MENU_ NEW = Menu.FIRST; 
protected final static int MENU DELETE = Menu.FIRST +1; 


(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.ex06 09 1); 


Bundle bunde = this.getIntent( ).getExtras(); 
path = bunde.getString("path"); 


Java.io.File file = new java.io.File(path); 
入 当 该 目录 不 存在 时 将 目录 创建 *%/ 
if (!file.exists()) 
{ 
file.mkdir(); 
} 
fill(file.listFiles()); 
} 


2) 定义 onOptionsItemSelected 方法 ， 根 据 单 击 的 菜单 实现 添加 或 删除 。 具 体 代 码 如 下 。 


(QOverride 
public boolean onOptionsItemSelected(Menultem item) 
{ 
super.onOptionsItemSelected(item); 
Switch (item.getItemId()) 
{ 
case MENU_NEW: 
上 # 单 击 添加 菜单 交 
showListActivity(path, ", ""); 
break; 
case MENU DELETE: 
入 单 击 删 除 沫 单 六 


deleteFile(); 
break; 

} 

return true; 


} 


3) 定义 onCreateOptionsMenu(Menu menu) 方 法 ， 用 于 添加 需要 的 菜单 项 。 具 体 代 码 
如 下 。 


(QOverride 
public boolean onCreateOptionsMenu(Menu menu) 


{ 
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super.onCreateOptionsMenu(menu); 

庆 添加 菜单 */ 

menu.add(Menu. NONE, MENU NEW, 0, R.string.strNewMenu); 
menu.add(Menu. NONE, MENU DELETE, 0, R.string.strDeleteMenu); 


return true; 
} 
4) 当 单 击 文件 名 后 取得 文件 内 容 ， 具 体 代 码 如 下 。 
@Override 


protected void onListItemClick 
(ListView1 View v, int position, long 1d) 
{ 
File file = new File 
(path + java.io.File.separator + items.get(position)); 


诺 单 击 文件 取得 文件 内 容 *%/ 
if (file.isFile()) 
{ 
String data = ""; 
try 
{ 
FileInputStream stream = new FileInputStream(file); 
StringBuffer sb = new StringBuffer(); 


int c; 
while ((c = stream.read()) != -1) 
{ 
sb.append((char) c); 
} 


stream.close(); 
data = sb.toString(); 
} 
catch (Exception e) 
{ 
e.printStackTrace(); 
} 
showListActivity(path, file.getName(), data); 
} 
} 


5) 定义 flFile[] files) 方 法 ， 用 于 填充 内 容 到 文件 。 具 体 代 人 码 如 下 。 


private void fll(File[] files) 
{ 


items =new ArrayList<String>(); 
if (files == null) 
{ 
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return; 


} 
for (File file : files) 
{ 
items.add(file.getName()); 
} 
ArrayAdapter<String> fileList = new ArrayAdapter<String> 
(this,android.R.layout.simple list item 1, items); 
setListAdapter(fileList); 
} 


6) 定义 showListActivity0)， 用 于 显示 已 经 存在 的 文件 列表 。 有 具体 代码 如 下 。 


private void showListActivity 

(String path, String Lilename, String data) 

{ 
Intent intent = new Intent(); 
intent.setClass(example8 1.this, example8 2.class); 


Bundle bundle = new Bundle(); 

大 全 
bundle.putString(“path”, path); 

/* 文件 | 

bundle.putString( [ilename”, [ilename); 
/* 文件 内 容 */ 

bundle.putString( “data”, data); 
intent.putExtras(bundle); 


startActivity(intent); 


} 
7) 定义 deleteFile0 方 法 ， 用 于 删除 选 定 的 文件 。 具 体 代码 如 下 所 示 : 


private void deleteFile() 


{ 


int position = this.getSelectedltemPosition(); 
if (position >= 0) 
{ 


File file = new File(path + java.io.File.separator 十 


| 


items.get(position)); 


谨 删除 文件 光 

file.delete(); 
items.remove(position); 
getListView().invalidateViews(); 
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3. 文件 example8_2.java 
当 单 击 “ 添 加 ”按钮 后 会 转 到 example8 2.java， 接 下 来 讲解 文件 example8 2.java， 其 具 
体 实现 流程 如 下 。 
1) 先 定义 三 个 变量 : myDialogEditText、MENU_SAVE 和 fileName。 具 体 代码 如 下 。 


public class example8 2 extends Activity 
{ 

private String path; 

private String data; 

private EditText myEditTextl; 


private EditText myDialogEditText; 
protected final static int MENU_ SAVE = Menu.FIRST; 
private String fileName; 


2) 设置 myEditTextl 用 于 放置 文件 内 容 ， 然 后 定义 Bundle 对 象 bunde， 用 于 获取 路 径 
和 数据 。 有 具体 代码 如 下 。 


(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.ex06 09 2); 


请 放置 文件 内 容 的 EditText */ 
myEditTextl = (EditText) fndViewById(R.id.myEditTextl); 


Bundle bunde = this.getIntent().getExtras(); 
path = bunde.getString("path"); 
data = bunde.getString("data"); 
fleName = bunde.getString("fileName"); 
myEditTextl.setText(data); 

} 


3) 使 用 onOptionsItemSelected 方法 根据 用 户 选择 而 进行 操作 。 当 选择 MENU_SAVE 
时 ,保存 这 个 文件 。 具 体 代码 如 下 。 


(QOverride 
public boolean onOptionsItemSelected(Menultem item) 
{ 
super.onOptionsItemSelected(item); 
Switch (item.getItemId()) 
{ 
case MENU SAVE: 
saveFile(); 
break; 
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} 


return true; 


4) 定义 onCreateOptionsMenu(Menu menu) 方 法 ， 用 于 添加 


(QOverride 
public boolean onCreateOptionsMenu(Menu menu) 


{ 


} 


5) 定义 saveFile0 方 法 ， 用 于 保存 文件 。 先 定义 LayoutInflater 对 象 factory， 用 于 跳出 存 
的 EditText， 最 后 通过 实 目 


人 码 如 下 。 
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档 。 然 后 通过 myDialogEditText 取得 Dialog ! 


super.onCreateOptionsMenu(menu); 
上 添加 菜单 * 
menu.add(Menu. NONE, MENU _ SAVE, 0, R.string.strSaveMenu); 


return true; 


private void saveFile() 


Hm 


谨 跳出 存档 的 对 话 框 六 
LayoutInflater factory = LayoutInflater.from(this); 


final View textEntryView = factory.inflate 
(R.layout.save_dialog, null); 


Builder mBuilderl = new AlertDialog.Builder(example8 2.this); 


mBuilderl.setView(textEntryView); 

旋 取得 对 话 框 里 的 EditText */ 

myDialogEditText = (EditText) textEntryView.findViewByld 
(R.id.myDialogEditText); 


myDialogEditText.setText(fileName); 


mBuilder1 .setPositiveButton 
( 
R.string.str_ alert ok,new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialoginterface, int 1) 
{ 
记得 肖 沁 


String Filename = path + java.io.File.separator 


个 荣 单 


。 有 具体 代码 如 下 。 


现存 档 处 理 


日 


体 代 


} 


} 
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+myDialogEditText.getText().toString(); 
Java.io.BufferedWriter bw; 
try 
{ 
bw = new java.io.BufferedWriter(new java.io.FileWriter( 
new java.io.File(Filename))); 
String str = myEditTextl.getText().toString(); 
bw.write(str, 0, str.length()); 
bw.newLine(); 
bw.close(); 
】 
catch (IOException e) 
{ 
e.printStackTrace(); 
} 
/* 回 到 example8 1*/ 
Intent intent = new Intent(); 
intent.setClass(example8 2.this, example8 1.class); 
Bundle bundle = new Bundle(); 
/* 将 路 径 传 到 example8_1 */ 
bundle.putString("path", path); 
intent.putExtras(bundle); 
startActivity(intent); 


finish(); 


mBuilderl .setNegativeButton(R.string.str_alert_cancel, null); 
mBuilderl1 .show(); 


执行 后 的 初始 效果 如 图 6-18 所 示 ， 当 单 击 一 个 按钮 后 会 显示 对 应 的 存储 信息 ， 如 图 6-19 


所 示 。 当 自 


[= 


examp 


SD Card 


图 6-18 初始 效果 


Bi 


le8 
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图 6-19 SD 卡 的 文件 信 ， 


上 图 6-19 中 的 “menu” 后 ， 会 弹出 两 个 menu 选项 ， 如 图 6-20 所 示 。 此 时 ， 可 
以 通过 这 两 个 选项 分 别 对 存储 卡 中 数据 实现 管理 。 
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可 以 添加 可 以 删除 


图 6-20 管理 menu 
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手机 闭 钟 ， 大 家 都 不 卫生 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 
Android 中 实现 定时 闻 钟 的 具体 过 程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\6\example9” 下 
面 开始 讲解 本 实例 的 具体 实现 流程 。 


6.9.1 ”实现 原理 

在 Android 中 提供 了 AlarmManager〔 闭 钟 组 件 ) 类 ， 通 过 此 类 可 以 设置 在 指定 时 间 运 行 
某 些 动 作 。 并 且 ， 在 Android 中 内 置 了 Alarm Clock〔 曾 钟 )， 通 过 AlarmManager 可 以 实现 
闸 钟 功能 。 


6.9.2 具体 实现 

本 实例 的 主 程序 文件 是 example9.java、example9 1.java 和 example9 2.java， 下 面 开始 
解 其 具体 实现 流程 。 

1. 文件 example9.java 

下 面 先 看 文件 example9.java， 其 具体 实现 流程 如 下 。 

1) 先 分 别 定义 6 个 变量 setTimel 、setTime2 、mButtonl 、mButton2、mButton3 及 
mButton4 ， 然 后 通过 findviewById 构造 3 个 对 象 myButton 、myProgressBar 和 
myTextView。 有 具体 代码 如 下 。 

public class example9 extends Activity 
{ 
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多 声明 变量 汐 
TextView setTime!l; 


TextView setTime2; 

Button mButton!1; 

Button mButton2; 

Button mButton3; 

Button mButton4; 

Calendar c=Calendar.getInstance(); 


2) 载 入 主 布局 文件 main.xml， 通 过 setTime10) 方 法 设置 闹钟 上 只 响 一 次 ， 然 后 设置 只 响 
次 闹钟 的 Buttonl 。 有 具体 代码 如 下 。 


(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
/六 载 入 main.xml Layout */ 
setContentView(R.layout.main); 


席 以 下 为 只 响 一 次 的 曾 钟 的 设置 *%/ 
setTimel=(TextView) fndViewById(R.id.setTimel); 
诺 只 响 一 次 的 益 钟 的 设置 Button */ 
mButton1=(Button)findViewById(R.idmButton1l); 
mButton1.setOnClickListener(new View.OnClickListener() 
{ 
public void onClick(View V) 
{ 
此 取得 单 击 按钮 时 的 时 间作 为 TimePickerDialog 的 默认 值 */ 
c.setTimeInMillis(System.currentTimeMillis()); 
int mHour=c.get(Calendar.HOUR OF DAY); 
int mMinute=c.get(Calendar.MINUTE); 


3) 通过 TimePickerDialog〔 时 间 对 话 框 来 弹出 一 个 对 话 框 ， 供 用 户 来 设置 时 间 。 
第 一 步 : 获取 设置 后 的 时 间 。 
第 二 步 : 指定 闹钟 设置 时 间 到 时 要 运行 CallAlarm.class。 

第 三 步 : 创建 PendingIntent。 

第 四 步 : 通过 AlarmManager.RTC_WAKEUP 设置 服务 在 系统 休眠 时 同样 会 运行 。 
第 五 步 : 定义 tmpS 对 象 ， 用 于 更 新 显示 的 闹钟 时 间 。 

第 六 步 : 以 Toast 提示 设置 已 完成 。 


具体 代码 如 下 。 


/# 跳出 TimePickerDialog 来 设置 时 间 */ 
new TimePickerDialog(example9.this， 


DOODDDOCDO 


new TimePickerDialog.OnTimeSetListener() 


{ 
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public void onTimeSet(TimePicker view,int hourOfDay, 


int minute) 


旋 取得 设置 后 的 时 间 ， 秒 和 毫秒 设 为 0 * 
c.setTimeInMillis(System.currentTimeMillis()); 
c.set(Calendar.HOUR OF DAY ,hourOfDay); 
c.set(Calendar.MINUTE,minute); 
c.set(Calendar.SECOND),0); 

c.set(Calendar. MILLISECOND.,0); 


/* 指定 闹钟 设置 时 间 到 时 要 运行 CallAlarm.class */ 
Intent intent = new Intent(example9.this, example9 2.class); 
/* 创建 PendingIntent */ 

PendingIntent sender=PendingIntent.getBroadcast( 


example9.this,0, intent, 0); 
/* AlarmManager.RTC_WAKEUP 设置 服务 在 系统 休眠 时 间 档 
* 以 set0 设 置 的 PendingIntent 只 会 运行 一 次 
* */ 


AlarmManager am; 
am=(AlarmManager)getSystemService(ALARM SERVICE); 
am.set(AlarmManager.RTC WAKEUP, 
c.getTimeInMillis(), 
sender 
); 
谨 更 新 显示 的 曾 钟 时 间 */ 
String tmpS=format(hourOfDay)+": "+format(minute); 
setTimel.setText(tmpS); 
/* 以 Toast 提示 设置 已 完成 */ 
Toast.makeText(example9.this," 设 置 曾 钟 时 间 为 "+tmpS， 
Toast.LENGTH SHORT) 
.Sshow(); 


ul 


| 
},mHour,mMinute,true).show(); 
} 
)); 


4) 设置 按钮 mButton2， 用 于 实现 只 响 一 次 的 闹钟 的 删除 。 

口 第 一 步 : 在 AlarmManager 中 实现 删除 。 

口 第 二 步 : 通过 Toast 提示 已 删除 设置 ， 并 更 新 显示 的 益 钟 时 间 。 
记 体 代码 如 下 。 


久 只 啊 一 次 的 闹钟 的 删除 Button *% 
mButton2=(Button) fndViewById(R.id.mButton2 ); 
mButton2.setOnClickListener(new View.OnClickListener() 


{ 
public void onClick(View V) 
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Intent intent = new Intent(example9.this, example9 2.class); 
PendingIntent sender=PendingIntent.getBroadcast( 
example9.this,0, intent, 0); 

/* 由 AlarmManager 中 删除 浆 

AlarmManager am; 

am =(AlarmManager)getSystemService(ALARM SERVICE); 

am.cancel(sender); 

雍 以 Toast 提示 已 误 除 设置 ， 并 更 新 显示 的 曾 钟 时 间 六 

Toast.makeText(example9.this," 闭 钟 时 间 解 除 "， 
Toast.LENGTH SHORT).show(); 

setTimel.setText(" 目 前 无 设置 "); 


重复 啊 起 
本 


: . 隔 秒 数 。 

第 四 步 : 获取 设置 的 开始 时 间 ， 秒 及 毫秒 都 设 为 0。 

第 五 步 : 指定 闹钟 设置 时 间 到 时 要 运行 CallAlarm.class。 
第 六 步 : 通过 setRepeating(0 方 法 让 闹钟 重复 运行 。 

第 七 步 : 通过 dmpS 更 新 显示 的 设置 曾 钟 时 间 。 

第 八 步 : 通过 以 Toast 提示 设置 已 完成 

人体 代 码 如 下 。 


詹 以 下 为 重复 响起 的 闹钟 的 设置 六 
setTime2=(TextView) findViewById(R.id.setTime2); 
雍 建立 重复 响起 的 闹钟 的 设置 画面 * 

/* 引用 timeset xml 为 Layout */ 

LayoutInflater factory = LayoutInflater.from(this); 


} 
); 
5) 开始 设置 重复 响起 的 闹钟 。 
第 一 步 : 建立 重复 响起 的 闹钟 的 设置 画面 ， 并 引用 timeset.xml 为 布局 文件 。 
第 二 步 : 建立 曾 钟 的 设置 Dialog 对 话 框 。 
汪 


DOODOOODDOCO 


final View setView = factory.inflate(R.layout.timeset,null); 

final TimePicker tPicker=(TimePicker)setView 
.findViewById(R.id.tPickem); 

tPicker.setIs24HourView(true); 


族 建立 重复 响起 六 钟 的 设置 Dialog */ 
final AlertDialog di=new AlertDialog.Builder(example9.this) 
.setIcon(R.drawable.clock) 
.setTitle(" 设 置 ") 
.SetView(setView) 
.SetPositiveButton(" 确 定 "， 
new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog, int which) 
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6) 实现 习 


{ 
雍 取得 设置 的 间隔 秒 数 尺 
EditText ed=(EditText)setView.findViewById(R.id.mEdit); 
int times=Integer.parseInt(ed.getText().toString()) 

*1000; 

铸 取得 设置 的 开始 时 间 ， 秒 及 毫秒 设 为 0 */ 
c.setTimeInMillis(System.currentTimeMillis()); 
c.set(Calendar.HOUR OF DAY,tPicker.getCurrentHour()); 
c.set(Calendar. MINUTE,tPicker.getCurrentMinute()); 
c.set(Calendar.SECOND.,0); 
c.set(Calendar.MILLISECOND.,0); 


庆 指定 闹钟 设置 时 间 到 时 要 运行 CallAlarm.class */ 
Intent intent = new Intent(example9.this， 


example9 2.class); 

PendingIntent sender = PendingIntent.getBroadcast( 
example9.this, 1, intent, 0); 

/* setRepeating() 方 法 可 让 曾 钟 重复 运行 */ 

AlarmManager am; 

am = (AlarmManager)getSystemService(ALARM SERVICE); 

am.setRepeating(AlarmManager.RTC WAKEUP, 

c.getTimeInMillis(),times,sender); 
入 更 新 显示 的 设置 闹钟 时 间 六 
String tmpS=format(tPicker.getCurrentHour())+": "+ 


format(tPicker.getCurrentMinute()); 
setTime2.setText(" 设 置 闹钟 时 间 为 "+tmpS+ 
"开始 ， 重 复 间隔 为 "ttimes/1000+" 秒 "); 
人 # 以 Toast 提示 设置 已 完成 *%/ 
Toast,makeText(example9.this," 设 置 闹钟 时 间 为 "HtmpS+ 
"开始 ， 重 复 间隔 为 "+times/1000+" 秒 ", 
Toast.LENGTH SHORT).show(); 


} 


.setNegativeButton(" 取 消 "， 
new DialogInterface.OnClickListener() 


public void onClick(DialogInterface dialog, int which) 


{ 
} 


}).create(); 


E 复 响起 的 闸 钟 的 设置 Button， 具 体 代码 如 下 。 


此 重复 响起 的 曾 钟 的 设置 Button 六 
mButton3=(Button) findViewById(R.id.mButton3); 
mButton3.setOnClickListener(new View.OnClickListener() 


{ 
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public void onClick(View v) 

{ 
庶 取得 单 击 按钮 时 的 时 间作 为 tPicker 的 默认 值 */ 
c.setTimeInMillis(System.currentTimeMillis()); 
tPicker.setCurrentHour(c.get(Calendar.HOUR OF DAY)); 
tPicker.setCurrent Minute(c.get(Calendar.MINUTE)); 
谍 跳出 设置 画面 di */ 
di.show(); 

} 

»); 


7) 开始 设置 重复 响起 的 闹钟 的 删除 Button。 

口 第 一 步 : 在 AlarmManager 中 删除 。 

口 第 二 步 : 以 Toast 提示 已 删除 设置 ， 并 更 新 显示 的 曾 钟 时 间 。 
人体 代码 如 下 。 


上 重复 响起 的 曾 钟 的 删除 Button 六 
mButton4=(Button) findViewById(R.id.mButton4); 
mButton4.setOnClickListener(new View.OnClickListener() 
{ 

public void onClick(View V) 

{ 


Intent intent = new Intent(example9.this, example9 2.class); 


PendingIntent sender = PendingIntent.getBroadcast( 
example9.this, 1, intent, 0); 

/* 由 AlarmManager 中 删除 *%/ 

AlarmManager am; 

am = (AlarmManager)getSystemService( ALARM SERVICE); 

am.cancel(sender); 

席 以 Toast 提示 已 删除 设置 ， 并 更 新 显示 的 闹钟 时 间 */ 

Toast.makeText(example9.this," 闭 钟 时 间 解 除 "， 
Toast.LENGTH SHORT).show(); 

setTime2.setText(" 目 前 无 设置 "); 


上 


8) 格式 设置 方法 format(int zx)， 用 于 设置 日 期 时 间 示 两 位 数 的 显示 格式 。 
族 目 期 时 间 显 示 两 位 数 的 方法 */ 
private String format(int x) 


{ 
String s=""+x; 
ifts.length()==1) s="0"+s; 
return s; 


具体 代码 如 下 。 
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} 
} 


2. 文件 example9_1.java 
下 面 先 看 文件 example9 1.java， 其 具体 实现 流程 如 下 。 
1) 加 载 相关 类 。 
2) 实现 实际 跳出 闹 铃 Dialog 的 Activity。 
3) 通过 AlertDialog.Builder(example9_ 1.this) 实 现 弹 出 闹钟 警示 框 。 
4) 通过 onClick(DialogInterface dialog, int whichButton) 关闭 Activity。 
去 体 代码 如 下 。 
入 加 载 相关 类 * 


import irdc.example9.R; 


import android.app.Activity; 
import android.app.AlertDialog; 
import android.content.DialogInterface; 


import android.os.Bundle; 


/x 实际 跳出 六 铃 Dialog 的 Activity */ 
public class example9 1 extends Activity 
{ 
(QOverride 
protected void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 

族 跳出 的 闹 铃 警示 。* 

new AlertDialog.Builder(example9 1.this) 
.SetIcon(R.drawable.clock) 
.setTitle(" 闸 钟 响 了 !1") 
.setMessage(" 赶 快 起 床 吧 !111") 
.SetPositiveButton(" 关 掉 他 "， 

new DialogInterface.OnClickListener() 


{ 


public void onClick(DialogInterface dialog, int whichButton) 
{ 

关闭 Activity */ 

example9 1.this.finish(); 


.Sshow(); 
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3. 文件 example9_2.java 
下 面 先 看 文件 example9 2.java， 其 具体 实现 流程 如 下 。 
1 ) 加 载 相 关 类 。 
2) 调用 闹钟 Alert 对 象 的 Receiver。 
3) 创建 mtent， 用 于 调用 AlarmAlert.class。 
号 体 代码 如 下 。 


package irdc.example9; 


入 加 载 相关 类 */ 

import android.content.Context; 

import android.content.Intent; 

import android.content.BroadcastReceiver; 


import android.os.Bundle; 


/* 调用 疮 钟 Alert 的 Receiver */ 
public class example9 2 extends BroadcastReceiver 


{ 
(QOverride 


public void onReceive(Context context, Intent intent) 


{ 


/* create Intent， 调 用 AlarmAlert.class */ 


Intent 1= new Intent(context, example9 1.class); 


Bundle bundleRet = new Bundle(); 
bundleRet.putString("STR CALLER",""); 
i.putExtras(bundleRet); 
i.addFlags(Intent.FLAG ACTIVITY NEW _ TASK); 
context.startActivity(i); 


} 


最 后 ， 还 需要 在 文件 AndroidManifest.xml 中 添加 对 CallAlarm 〈 呼 叫 闹 钟 ) 的 receiver 
设置 。 具体 代码 如 下 。 


<! 一 注册 receiver CallAlarm --> 


<receiver android:name=".example9 2" android:process=":remote" /> 
<activity android:name=".example9 1" ndroid:label="(@string/app_name"> 


</activity> 


过 


bia me 


执行 后 的 初始 效果 如 图 6-21 所 示 ， 单 击 第 一 个 “设置 ”按钮 后 在 弹出 的 界面 中 可 以 设 
置 闹钟 时 间 ， 如 图 6-22 所 示 。 单 击 第 二 个 按钮 可 以 设置 重复 响起 的 时 间 ， 如 图 6-23 所 示 。 
闹钟 响起 后 的 界面 如 图 6-24 所 示 。 
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碟 天 全 3:50Aw 
example9 
3:50:14 am 
响 一 次 [设置 | 
没 设置 删除 
重复 响 设置 
没 设置 删除 


图 6-21 初始 效果 


念 设置 


Fria 


03 


图 6-23 重复 响 的 设置 界面 


| 由 


6a 四 


几乎 每 个 手机 中 都 有 黑 名 单 功 能 ， 被 列 入 黑 名 单 的 月 
节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 实现 
体 过 程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\6\example10”， 下 面 开 


现 流程 。 
6.10.1 ”实现 原理 
在 本 实例 中 ， 首 先 添加 一 个 EditText， 月 


” Android 开发 完全 实战 宝典 


Set Cancel 


图 6-22” 啊 一 次 的 设置 界面 


合 闹钟 响 了 1! 


赶快 起 床 吧 凯 


关 掉 他 


旧 户 在 里 面 可 以 输入 黑 名 单 月 


图 6-24 ” 疗 钟 响起 后 界面 


上 户 不 能 打 进 电话 和 发 送 短信 。 在 本 
名 单 用户 来 电 静 音 的 具 
6 讲解 本 实例 的 具体 实 


昌 户 的 号 码 。 当 此 号 


码 来 电 时 ， 系 统 会 自动 切换 为 静音 模式 。 当 对 方 挂机 后 ， 再 自动 转换 为 正常 模式 ， 并 使 用 


Toast 实现 提示 。 


在 具体 实现 上 ， 转 换 铃 声 模 式 功 能 是 通过 setRingerMode 参数 实现 的 ， 正 常 模 式 是 


RINGER MODE NORMAL ， 静音 模式 是 RINGER MODE SILENT ， 震 动 模式 是 


RINGER MODE VIBRATE。 
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6.10.2 具体 实现 


本 实例 的 主 程序 文件 是 example10.java， 下 面 开 始 讲 解 其 具体 实现 流程 。 


HO 


1) 先 分 别 定 义 3 个 变量 mTextView01、mEditTextl 和 mEditText03。 具 体 代 码 如 下 。 


public class example10 extends Activity 


{ 


private TextView mTextView01; 


private TextView mTextView03; 
private EditText mEditTextl; 


2) 设置 PhoneCallListener 对 象 phoneListener， 用 TelephonyManager 抓 取 Telephony 
Severice， 然 后 设置 Listen Call0 方 法 ， 并 查找 TextView 和 EditText 的 数据 。 有 具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


诺 设 置 PhoneCallListener*/ 
mPhoneCallListener phoneListener=new mPhoneCallListener(); 


和 


SS 


用 TelephonyManager 抓 取 Telephony Severice*/ 


TelephonyManager telMgr = (TelephonyManager)getSystemService( 


TELEPHONY _ SERVICE); 


诺 设 置 Listen Call*/ 
telMgr.listen(phoneListener, mPhoneCallListener. 


LISTEN CALL STATE); 


/# 查 找 TextView 和 EditText 中 的 数据 */ 

ImTextView01l = (TextView)findViewById(R.id.myTextViewl); 
ImTextView03 = (TextView)findViewById(R.id.myTextView3); 
mEditTextl = (EditText)findViewById(R.id.myEditTextl ); 


3) 判断 PhoneStateListener 当前 状态 。 


} 
口 第 一 步 : 
口 第 二 步 : t 
口 第 三 步 : 
口 第 四 步 : 
口 第 五 步 : 


获取 手机 符 机 状态 。 
设置 手机 为 待机 时 响 铃 正常 。 

如 果 获 取 手 机 状态 为 通话 中 则 显示 对 应 信息 。 

如 果 获 取 手 机 状态 为 来 电 则 显示 来 电信 息 。 

判断 输入 电话 是 否 一 致 ， 如 果 一 致 时 用 静音 。 


L 体 代码 如 下 。 
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/# 判断 PhoneStateListener 当前 状态 */ 
public class mPhoneCallListener extends PhoneStateListener 
{ 
(QOverride 
public void onCallStateChanged(int state, String incomingNumber) 
{ 
switch(state) 
{ 
入 获取 手机 待机 状态 光 
case TelephonyManager.CALL STATE IDLE: 
mTextView01.setText(R.string.str CALL STATE IDLE); 


try 
{ 
AudioManager audioManager = (AudioManager) 
getSystemService(Context.AUDIO SERVICE); 
if (audioManager != null) 
{ 
虚设 置 手机 为 待机 时 响 铃 正 常 */ 
audioManager.setRingerMode(AudioManager. 
RINGER MODE NORMAL); 
audioManager.getStreamVolume( 
AudioManager.STREAM RING); 


} 


catch(Exception e) 

{ 
mTextView01.setText(e.toString()); 
e.printStack Trace(); 


} 
break; 


雍 获取 手机 状态 为 通话 中 将 

case TelephonyManager.CALL STATE OFFHOOK: 
mTextView01.setText(R.string.str CALL STATE OFFHOOR); 
break; 


席 ”获取 手机 状态 为 来 电 

case TelephonyManager.CALL STATE RINGING: 
入 显示 来 电信 息 *% 
ImTextView01.SetText( 


getResources().getText(R.string.str CALL STATE RINGINGO)+ 


incomingNumber); 


A 判断 给 入 电话 是 否 一 致 ， 一 致 时 用 静音 
if(incomingNumber.equals(mTextView03.getText().toString())) 
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{ 
try 
{ 
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AudioManager audioManager = (AudioManager) 
getSystemService(Context.AUDIO SERVICE); 


if (audioManager != null) 


{ 
/# 设 置 响 铃 为 静音 */ 


audioManager.setRingerMode(AudioManager. 


RINGER MODE SILENT); 
audioManager.getStreamVolume( 


AudioManager.STREAM RING); 
Toast.makeText(examplel10.this, getString(R.string.str msg) 
,Toast.LENGTH SHORT).show(); 


} 

catch(Exception e) 

{ 
mTextView01.setText(e.toString()); 
e.printStack Trace(); 

break; 


} 


super.onCallStateChanged(state, incoming Number); 


mbEditTextl .setOnKeyListener(new EditText.OnKeyListener() 


{ 


示 在 TextView 对 象 中 。 具 体 代 码 如 下 。 


执行 后 


(QOverride 


public boolean onKey(View v, int keyCode, KeyEvent event) 


{ 
记 设 置 EditText 的 输入 数据 显示 在 


TextView*/ 


mTextView03.setText(mEditTextl .getText()); 


return false; 


)); 


的 初始 效果 如 图 6-25 所 示 ， 在 输入 机 


当 此 号 码 来 9 


有 时 会 是 静音 模式 ， 如 图 6-27 所 示 。 


EE 中 可 以 输入 时 


上 名 单 号 码 ， 


于 将 EditText 的 输入 数据 显 


如 图 6-26 所 示 。 
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视 剧 全 3:59Aw 


example10 


example10 


上 :于 待机 状态 


处 于 待机 状 


当前 久 


13488888888 


6-25 ”初始 效果 


国友 出 人 @ 4:00 ww 


13488888888 
电话 号 是 48888888 


| 


当前 配音 模式 .… 


ALT 


图 静音 


6-27 来 


注意 ;上述 实例 在 模拟 器 中 不 会 显 


加 


在 移动 手机 设备 中 ， 通 常 为 用 户 提供 了 在 不 同时 间 显 示 不 同 屏幕 背景 照片 的 功能 


示 静 音 模式 图 片 ， 但 是 在 真实 机 器 上 会 显示 ， 


。 在 本 


节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 指定 时 间 置 换 桌 面 背 景 的 具体 过 程 。 
TT 始 讲 解 本 实例 的 其 体 实现 流程 。 


例 的 源 代码 保存 在 “光盘 :\daimaN6\example11”， 下 面 3 
6.11.1 ”实现 原理 


月 


本 实 


昌 作 曾 钟 ， 它 还 可 


本 实例 结合 了 前 面 的 闹钟 实例 ， 实 际 上 AlarmManage 类 并 不 是 只 能 
置 在 什么 时 间 运 行 什么 样 的 动作 。 


图 片 供 月 


行 设 


在 本 实例 中 ， 预 先 准备 了 7 张 素材 
6.11.2 具体 实现 


以 自 


日 户 选 择 ， 放 在 了 “res\drawable” 目 录 下 。 


本 实例 的 主 程序 文件 是 examplell.java、ChangeBgImage.java 、DailyBgDB.java 和 


解 其 


具体 实现 流程 。 


下 


鳃 玫 


MyReceiver.java, F 始 讲 
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1. 文件 example11.java 
下 面 先 看 文件 examplell.java， 其 具体 实现 流程 如 下 。 


1) 先 加 载 相关 类 ， 然 后 声明 设置 
Button 变量 ， 并 声明 显示 7 个 图 文件 名 称 的 TextView 变量 。 有 具体 代码 如 下 。 


现 


吧 


public class examplel 1 extends Activity 


{ 
旋 声明 设置 图 片 的 七 个 Button 及 启动 与 终止 的 两 个 Button */ 
private Button mButton1; 


private Button mButton2; 

private Button setButton1; 

private Button setButton2; 

private Button setButton3; 

private Button setButton4; 

private Button SetButton5; 

private Button setButton6; 

private Button setButton 7; 

伴 声明 显示 图 文件 名 称 的 七 个 TextView */ 
private TextView mySetl; 


private TextView mySet2; 
private TextView mySet3; 
private TextView mySet4; 
private TextView mySetS; 
private TextView mySet6; 
private TextView mySet’7; 


2) 分 别 声明 自 定义 的 数据 库 变 量 DailyBgDB， 存 放 设置 值 的 Map， 存 放 医 


4 


组 bg 与 存放 图 文件 名 称 的 数组 bgName， 具 体 代码 如 下 。 
居 声明 自 定义 的 数据 库 变量 DailyBgDB */ 
private DailyBgDB db; 
从 声明 存放 设置 值 的 Map */ 
private Map<Integer,Integer> map; 


private LayoutInflater inflater; 

private int tmp Which=0; 

从 声明 存放 图 文件 id 的 数组 bg 与 存放 图 文件 名 称 的 数组 bgName */ 
private final int[] bg = 
{R.drawable.b01,R.drawable.b02,R.drawable.b03,R.drawable.b04, 
R.drawable.b05,R.drawable.b06,R.drawable.b07}; 

private final String[] bgName = 
{"b01.png","b02.png","b03.png","b04.png","b05.png","b06.png", 
"b07.png"}; 


片 的 7 个 Button 按钮 变量 及 启动 与 终止 的 两 个 


文件 id 的 数 


3) 载 入 主 布局 文件 main.xml， 并 将 数据 库存 放 的 设置 值 放 入 map 中 ， 然 后 初始 化 各 个 


TextView 对 象 ， 具 体 代码 如 下 。 
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4) 根据 获取 的 图 像 设置 显示 的 图 文件 名 称 ， 具 
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(QOverride 
public void onCreate(Bundle savedInstanceState) { 
super.onCreate(savedInstanceState); 
/六 载 入 main.xml Layout */ 
setContentView(R.layout.main); 
inflater=(LayoutInflater)getSystemService( 
Context.LAYOUT INFLATER SERVICE); 


上 将 数据 库存 放 的 设置 值 放 入 map 中 */ 
initSettingData(); 
/# 初始 化 TextView 对 象 */ 

mySetl=(TextView) findViewById(R.id.mySetl); 
mySet2=(TextView) findViewById(R.id.mySet2); 
mySet3=(TextView) findViewByld(R.id.mySet3); 
mySet4=(TextView) findViewByld(R.id.mySet4); 
mySetS=(TextView) findViewByld(R.id.mySet5); 
mySet6=(TextView) findViewByld(R.id.mySet6); 
mySet7=(TextView) findViewById(R.id.mySet7); 


体 代 码 如 下 。 


雍 设置 显示 的 图 文件 名 称 六 
if(!Imap.get(0).equals(99)) 
{ 

mySetl.setText(bgName[map.get(0)]); 
} 
if(!Imap.get(1).equals(99)) 
{ 

mySet2.setText(bgName[map.get(1)]); 
} 
if(!Imap.get(2).equals(99)) 
{ 

mySet3.setText(bgName[map.get(2)]); 
} 
if(!Imap.get(3).equals(99)) 
{ 

mySet4.setText(bgName[map.get(3)]); 
} 
if(!Imap.get(4).equals(99)) 
{ 

mySetS.setText(bgName[map.get(4)]); 
} 
if(!Imap.get(5).equals(99)) 


{ 
mySet6.setText(bgName[map.get($)]); 


} 
if(!Imap.get(6).equals(99)) 


5) 初始 化 各 个 Button 对 象 ， 然 后 以 initButton() 方 法 来 设置 OnClickListener， 


如 下 。 


) 


CN 


DOODODDOCDO 
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mySet7.setText(bgName[map.get(6)]); 


} 


许 初始 化 Button 对 象 */ 

setButton1=(Button) findViewById(R.id.setButton!1); 
setButton2=(Button) fmdViewById(R.id.setButton2); 
setButton3=(Button) fmdViewById(R.id.setButton3); 
setButton4=(Button) fmdViewById(R.id.setButton4); 
setButton5=(Button) findViewBylId(R.id.setButton5$); 
setButton6=(Button) fmdViewById(R.id.setButton0); 
setButton7=(Button) fmdViewById(R.id.setButton7); 
/* 以 initButton0 来 设置 OnClickListener */ 
setButton1=initButton(setButton1,mysSet1,0); 
setButton2=initButton(setButton2,mysSet2,1); 
setButton3=initButton(setButton3,mySet3,2); 
setButton4=initButton(setButton4,mySet4,3); 
setButton$=initButton(setButton$,mySetS,4); 
setButton6=initButton(setButton6,mySet6,5); 
setButton7=initButton(setButton7,mySet7,6); 


设置 启动 服务 的 Button， 通 过 setOnClickListener 实现 单 击 事件 的 监听 。 


AAA 


全 一 上 


第 一 步 : 取得 服务 启动 后 一 天 的 0 点 0 分 0 秒 的 millsTime。 


第 二 步 : 重复 运行 的 间隔 时 间 。 


第 三 步 : 将 更 换 背 景 的 进程 添加 AlarmManager 中 。 
寸 setRepeating(0) 方 法 让 进程 处 理 重 复 运行 。 


本 
避 


第 四 步 : 


第 五 步 : 使 用 Toast 提示 已 启动 。 


第 六 步 : 启动 后 蕊 上 先 运行 一 次 换 背 景 的 程序 以 更 换 今天 的 背景 。 


体 代 码 如 下 。 


~ 


雍 设置 局 动 服 务 的 Button */ 
mButton1=(Button)findViewById(R.id.myButton1); 
mButton1.setOnClickListener(new View.OnClickListener() 


{ 


public void onClick(View V) 


{ 
族 取得 服务 


启动 后 一 天 的 0 点 0 分 0 秒 的 millsTime */ 


Calendar calendar=Calendar.getInstance(); 
calendar.add(Calendar.DATE,1); 
calendar.set(Calendar.HOUR OF DAY,0); 
calendar.set(Calendar.MINUTE.,0); 
calendar.set(Calendar.SECOND,0); 
calendar.set(Calendar.MILLISECOND,0); 


号 体 代码 
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long startTime=calendar.getTimeInMillis(); 
族 重复 运行 的 间隔 时 间 */ 
long repeatTime=24*60*60*1000; 
访 将 更 换 桌 布 的 进程 添加 AlarmManager 中 */ 
Intent intent = new Intent(examplel1.this,MyReceiver.class); 


PendingIntent sender = PendingIntent.getBroadcast( 
examplell.this, 0, intent, 0); 
AlarmManager am = (AlarmManager)getSystemService( 
ALARM SERVICE); 

/* setRepeating() 可 让 进程 重复 运行 

startTime 为 开始 运行 时 间 

repeatTime 为 重复 运行 间隔 

AlarmManager.RTC 可 使 服务 休眠 时 仍然 会 运行 */ 
am.setRepeating(Alarm Manager.RTC,startTime,repeatTime, 


sender); 
上 六 以 Toast 提示 已 启动 */ 
Toast.makeText(examplel11.this," 服 务 已 启动 ",Toast.LENGTH_SHORT) 
.Show(); 
雍 启动 后 马上 先 运行 一 次 换 背 景 的 程序 以 更 换 今天 的 背景 % 
Intenti= new Intent(examplell.this,ChangeBgImage.class); 
startActivity(2); 
} 
D); 


件 的 监听 。 


0 
| 


7) 设置 终止 服务 的 Button， 定 义 onClick(View 由) 方法 实现 对 单 击 按 钾 习 
口 定义 Intent 对 象 intent。 
口 在 AlarmManager 中 删除 调度 。 

口 通过 Toast 提示 已 终止 。 

纪 体 代码 如 下 。 


上 # 设置 终止 服务 的 Button */ 
mButton2=(Button) findViewById(R.id.myButton2); 
mButton2.setOnClickListener(new View.OnClickListener() 
{ 

public void onClick(View v) 


{ 


Intent intent = new Intent(examplel1.this,MyReceiver.class); 


PendingIntent sender = PendingIntent.getBroadcast( 
examplel1.this, 0, intent, 0); 
愉 由 AlarmManager 中 删除 调度 */ 
AlarmManager am = (AlarmManager)getSystemService( 
ALARM SERVICE); 
am.cancel(sender); 
/# 通过 Toast 提示 已 终止 *%/ 
Toast makeText(examplell.this," 服 务 已 终 上 ",ToastLENGTH SHORT) 
.Show0O; 
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); 


8) 定义 initSettingData0， 用 于 从 数据 库 中 取得 设置 值 的 方法 


入 由 数据 库 中 取得 设置 值 的 方法 */ 
private void initSettingData() 


{ 
map=new LinkedHash Map<Integer,Integer>(); 


db=new DailyBgDB(examplell.this); 
Cursor cur=db.select(); 


while(cur.moveToNext()){ 
map.put(cur.getInt(0),cur.getInt(1)); 

} 

cur.close(); 

db.close(); 


o 


具体 代码 如 下 。 


9) 设置 Button 的 OnClickListener 的 方法 ， 首 先 设置 单 击 Button 后 跳出 的 选择 
Dialog， 然 后 设置 预览 画面 的 文件 名 与 InageView (图片 组 件 )， 最 后 改变 画面 显示 的 图 文件 


文件 名 ， 并 将 更 改 的 信息 存 入 数据 库 。 有 具体 代码 如 下 。 


/# 设置 Button 的 OnClickListener 的 方法 */ 
private Button initButton(Button b,final TextView tbfinal int 1d) 
{ 
b.setOnClickListener(new View.OnClickListener() 
{ 
public void onClick(View V) 
{ 
/# 设置 单 击 Button 后 跳出 的 选择 图 片 的 Dialog */ 
new AlertDialog.Builder(examplel1.this) 


.SetIcon(R.drawable.pic icom) 
.SetTitle(" 请 选择 图 片 ! ") 
.SetSingleChoiceItems(bgName,map.get(id)， 


new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog,int which) 


{ 
tmpWhich=which; 
雍 选择 图 片 后 跳出 预览 图 文件 的 窗口 光 


View view=inflater.inflate(R.layout.preview, null); 


TextView message=(TextView) view.findViewById( 


R.id.bgName); 
旋 设置 预览 画面 的 文件 名 与 InageView */ 
Imessage.SetText(bgName[which]); 


ImageView mView01 = (ImageView)view.findViewById( 


名 


片 的 
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R.id.bgImage); 
mView01.setImageResource(bg[which]); 
Toast toast=Toast.makeText(examplel1 .this,"", 
Toast.LENGTH SHORT); 
toast.setView(view); 
toast.show(); 
} 
D) 
.SetPositiveButton(" 确 定 "， 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog,int which1) 
{ 
放 改变 画面 显示 的 设置 图 文件 文件 名 */ 
tsetText(bgName[tmpWhich]); 
/* 改变 map 对 象 里 的 值 */ 
map.put(id,tmp Which); 
族 将 更 改 的 设置 存 入 数据 库 * 
saveData(id,tmpWhich); 
} 
DD) 


.setNegativeButton(" 取 消 "， 
new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog,int which) 
{ 
} 
}).show(); 
} 
»: 
return b; 


10) 定义 saveData(int id,int value) 方 法 ， 用 于 将 设置 值 存储 到 数据 库 。 具 体 代码 如 下 。 


族 存储 设置 值 至 DB 的 方法 */ 

private void saveData(int id,int value) 

{ 
db=new DailyBgDB(examplell.this); 
db.update(id,value); 
db.close(); 

) 

} 


| 
Tr 


2. 文件 MyReceiver.java 
接 下 来 看 文件 MyReceiverjava， 其 功能 是 引入 相关 类 ， 然 后 运行 更 换 桌 面 背 景 的 
290O 国 国 


Receiver。 


L 体 代码 如 下 。 


package irdc.examplell; 


入 加 载 相关 类 */ 

import android.content.Context; 

import android.content. Intent; 

import android.content.BroadcastReceiver; 
import android.os.Bundle; 


/* 运行 更 换 桌 面 背景 的 Receiver */ 
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public class MyReceiver extends BroadcastReceiver 


{ 
(@Override 


public void onReceive(Context context, Intent intent) 


{ 


/* create Intent， 调 用 ChangeBgImasge.class */ 


Intent 1= new Intent(context, ChangeBgImage.class); 


Bundle bundleRet = new Bundle(); 


bundleRet.putString("STR CALLER", ""); 


i.putExtras(bundleRet); 


i.addFlags(Intent.FLAG ACTIVITY NEW _ TASK); 


context.startActivity(i); 


} 


3. 文件 ChangeBglmage.java 


接 下 


来 看 文件 ChangeBgImage.java， 其 具 


1) 图 


E 加 载 相关 类 ， 具 体 代 码 如 下 。 


package irdc.examplell; 
/* import 相关 class */ 
import irdc.examplel1.R; 


import java.io.IOException 

import java.util.Calendar; 

import android.app.Activity; 

import android.database.Cursor; 

import android.graphics.Bitmap; 

import android.graphics.BitmapFactory; 
import android.os.Bundle; 


2) 实际 运行 更 换 桌 面 背景 的 Acetivity， 


如 下 。 


体 实现 流程 如 下 。 


声明 存放 图 文人 


F id 的 数组 bg， 


体 代码 
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习 实际 运行 更 换 桌 面 背景 的 Activity */ 
public class ChangeBgImage extends Activity 
{ 
具 声明 存放 图 文件 id 的 数组 bg */ 
private static final int[] bg = 
{R.drawable.b01,R.drawable.b02,R.drawable.b03,R.drawable.b04, 
R.drawable.b05,R.drawable.b06,R.drawable.b07}; 


3) 载 入 布局 文件 progress.xml， 并 获取 今天 是 星期 几 ， 然 后 从 数据 库 中 取得 今天 应 该 换 
哪 一 张 背 景 ， 如 果 “DailyBg==99” 代 表 没 设置 ， 则 不 运行 。 具 体 代码 如 下 。 
(QOverride 


protected void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


J 2 


/# 载 入 progress.xml Layout */ 
setContentView(R.layout.progress); 

访 取得 今天 是 星期 几 */ 

Calendar ca=Calendar.getInstance(); 

int dayOf Week=ca.get(Calendar.DAY OF WEEK)-1; 


雍 从 数据 库 中 取得 今天 应 该 换 哪 一 张 背景 */ 

int DailyBg=0; 

String selection = "Dailyld=?"; 

String[] selectionArgs = new String[]{""+dayOfWeek}; 
DailyBgDB db=new DailyBgDB(ChangeBgImage.this); 
Cursor cur=db.select(selection,selectionArgs); 


while(cur.moveToNext()) 
{ 

DailyBg=cur.getInt(0); 
) 
cur.close(); 
db.close(); 


/* 如 果 DailyBg==99 代表 没 设置 ， 所 以 不 运行 */ 
if(DailyBg!=99) 
{ 
Bitmap bmp=BitmapFactory.decodeResource 
(getResources(), bg[DailyBg]); 
try 
{ 
super.setWallpaper(bmp); 
} 
catch (IOException e) 
{ 
e.printStack Trace(); 


} 
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} 
finish(); 
} 
} 


4. 文件 DailyBgDB.java 
接 下 来 看 文件 DailyBgDB.java， 其 具体 实现 流程 如 下 。 


1) 图 


package irdc.examplell; 


入 加 载 相关 类 */ 

import android.content.ContentValues; 

import android.content.Context; 

import android.database.Cursor; 

import android.database.sqlite.SQLiteDatabase; 
import android.database.sqlite.SQLiteOpenHelper; 


public class DailyBgDB extends SQLiteOpenHelper 
{ 
化 变量 声明 * 
private final static String DATABASE NAME = "dailyBG db"; 
private final static int DATABASE VERSION= 1; 
private final static String TABLE NAME = "dailySetting table"; 
public final static String FIELD1 = "Dailyld"; 
public final static String FIELD2 = "DailyBg"; 
public SQLiteDatabase sdb; 


族 构造 器 */ 
public DailyBgDB(Context context) 
{ 


super(context, DATABASE NAME, null, DATABASE VERSION); 


sdb= this.getWritableDatabase(); 


2) 3 
体 代码 如 


| 
和 T 始 数据 库 处 理 ， 如 果 表 格 不 存在 就 创建 表 对 象 ， 并 存 入 初始 的 数据 到 数据 库 。 具 
下 。 
(QOverride 
public void onCreate(SQLiteDatabase db) 
{ 


人 * 表 不 存在 就 创建 表 */ 


人 体 代 码 如 下 。 


进行 变量 声明 ， 然 后 定义 构造 器 DailyBgDB(Context context)。 


String sql = "CREATE TABLE IF NOT EXISTS "+TABLE NAME+"("+FIELD!1 


+" INTEGER primary key, "+FIELD2+" INTEGER)'"; 
db.execSQL(sq}l); 


庶 存 入 初始 的 数据 到 数据 库 */ 
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sdb=db; 

insert(0,99); 
insert(1,99); 
insert(2,99); 
insert(3,99); 
insert(4,99); 
insert($,99); 
insert(6,99); 


(QOverride 
public void onUpgrade(SQLiteDatabase db,int oldV ersion, 


int newVersion) 


public Cursor select() 
{ 
Cursor cursor=sdb.query(TABLE NAME,null,null, 
null,null,null,null); 


return cursor; 


} 


3) 定义 方法 select(String selection,String[] selectionArgs)， 当 select 时 ， 


痊 


此 方法 ， 用 于 检索 数据 。 具 体 代 码 如 下 。 


/* select 时 有 where 条 件 要 用 此 方法 */ 
public Cursor select(String selection, String[] selectionArgs) 


{ 
String[] columns = new String[] { FIELD2 }; 


Cursor cursor=sdb.query(TABLE NAME,columns,selection,selectionArgs,null,null,null); 


return cursor; 


} 


于 将 添加 的 值 放 入 ContentValues。 


ME 


4) 定义 方法 insert(int valuel,int value2)， 


如 下 。 


public long insert(int valuel,int value2) 

{ 
诺 将 添加 的 值 放 入 ContentValues */ 
ContentValues cv = new ContentValues(); 
cv.put(FIELD!1, valuel); 
cv.put(FIELD2, value2); 
long row = sdb.insert(TABLE NAME, null, ¢v); 
return row; 
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有 where 条 件 


妹 


闷 体 代码 
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。 具体 代码 如 下 。 


ml 


5) 定义 方法 delete(int id)， 用 于 删除 设 


public void delete(int id) 

{ 
String where = FIELD1 + "= ?2"; 
String[] whereValue ={ Integer.toString(id) }; 
sdb.delete(TABLE NAME, where, whereValue); 


} 
6) 定义 update(int id, int value) 方 法 ， 用 于 修改 设置 。 具 体 代码 如 下 。 


public void update(int id, int value) 

{ 
String where = FIELD1 + "= ?7"; 
String[] whereValue ={ Integer.toString(id) }; 
上 将 修改 的 值 放 入 ContentValues */ 
ContentValues cv = new ContentValues(); 
cv.put(FIELD2, value); 
sdb.update(TABLE NAME, cv, where, whereValue); 

} 

} 


最 后 还 需要 在 文件 AndroidManifest.xml 中 添加 MyReceiver 的 receiver 设置 ， 并 添加 背 
景 图 像 权限 android.permission.SET WALLPAPER 。 有 具体 代码 如 下 。 


<receiver android:name=".MyReceiver" android:process=":remote'"/> 
<I-- 设 定 SET WALLPAPER 权限 --> 
<uses-permission android:name="android.permission.SET WALLPAPER" /> 


执行 后 的 初始 效果 如 图 6-28 所 示 ， 选 择 星期 数 ， 单 击 后 面 的 “设置 ”按钮 后 ， 可 以 在 
弹出 界面 中 设置 这 天 的 背景 图 片 ， 如 图 6-29 所 示 。 


Lm 国 态 间 龟 4:08m 吗 国名 而 拓 4:09 
加 请 先 ! 
每 天 自动 换 背 景 请 选择 图 片 
b03.png 
星期 7 : 当前 无 设置 | 设置 | 
星期 1 : ”当前 无 设置 设置 b04.png 
星期 2 : 当前 无 设置 设置 
星期 3 : 当前 无 设置 设置 b05.png 
星期 4 : 当前 无 设置 设置 bOG. prig 
星期 5 : ”当前 无 设置 设置 
星期 6 : ”当前 无 设置 设置 b07.png 
[La | | | 


图 6-28 初始 效果 
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2 寺 四 和 四 本 四 晶 


当 发 送 短信 后 ， 用 户 往 往 对 是 否 发 送 成 功 比 较 关 心 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 
实例 的 实现 ， 介 绍 监听 短信 发 送 的 具体 过 程 。 本 实例 的 源 代 码 保存 在 “光盘 :daimaN6 
\example12”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 


o 


6.12.1 ”实现 原理 
本 实例 中 ， 当 发 送 一 条 短信 后 ， 会 及 时 提供 一 条 信息 ， 说 明 短 信 是 发 送 成 功 还 是 失败 。 
手机 的 默认 程序 可 以 捕捉 到 发 送 状 态 ， 这 是 因为 经 过 系统 广播 的 信息 ， 程 序 可 以 捕捉 到 发 送 
结果 。 本 实例 的 学 习 重 点 是 如 何 衍 生 广 播 类 mServiceReceiver， 并 在 这 个 Receiver 对 象 中 判 
断 短信 的 发 送 结果 。 
6.12.2 具体 实现 
本 实例 的 主 程序 文件 是 example12.java， 下 面 开 始 讲解 其 具体 实现 流程 。 


1)〉 先 创建 两 个 mServiceReceiver 对 象 ， 作 为 类 成 员 变 量 。 然 后 分 别 创建 5 个 变量 : 
mButton1、mButton2、mTextView01、mEditTextl 和 mEditText2。 上 有 具体 代码 如 下 。 


由 


public class example12 extends Activity 


/# 创建 两 个 mServiceReceiver 对 象 ， 作 为 类 成 员 变量 */ 
Private mServiceReceiver mReceiver01, mReceiver02; 


private Button mButton1; 
private TextView mTextView0!1; 
private EditText mEditTextl, mEditText2; 


2) 自 定义 ACTION 常数 ， 作 为 广播 的 Intent Filter 识别 常数 。 然 后 通过 mEditTextl 获取 
电话 号 码 ， 通 过 mEditText2 对 象 获取 信息 内 容 ， 然 后 设置 默认 值 为 5556， 表 示 第 二 个 模拟 
器 的 Port。 有 具体 代码 如 下 。 

/# 自 定 义 ACTION 常数 ， 作 为 广播 的 Intent Filter 识别 常数 */ 
private String SMS SEND ACTIOIN= "SMS SEND ACTIOIN"; 
private String SMS DELIVERED ACTION= "SMS DELIVERED ACTION "; 


(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mTlextView01 = (TextView)findViewBylId(R.id.myTextView!1); 


雍 电话 号 码 */ 
mbEditTextl = (EditText) findViewById(R.id.myEditText1); 
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入 短信 和 内容 */ 
mEditText2 = (EditText) findViewById(R.id.myEditText2); 
mButton1 = (Button) fndViewById(R.id.myButton1l); 


//mEditTextl .setText("+12345678"); 

入 设置 默认 值 为 5556 表示 第 二 个 模拟 器 的 Port */ 
mEditTextl.setText("5556"); 
mbEditText2.setText("Hello DavidLanz!"); 


3) 定义 setOnClickListener 作为 发 送 SMS 短信 按钮 事件 处 理 程序 ，strDestAddress 对 象 
是 欲 发 送 的 电话 号 码 ，strMessage 对 象 是 要 发 送 的 内 容 。 具 体 代 码 如 下 。 


席 发 送 SMS 短信 按钮 事件 处 理 */ 
mButton1.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 


public void onClick(View arg0) 
{ 
/TODO Auto-generated method stub 


铸 欲 发 送 的 电话 号 码 党 
String strDestAddress = mEditTextl.getText().toString(); 


雍 欲 发 送 的 短信 内 容 沁 
String strMessage = mEditText2.getText().toString(); 


4) 创建 SmsManager 对 象 smsManager， 用 于 发 送 短信 。 有 具体 流程 如 下 。 
第 一 步 : 创建 自 定义 Action 常数 的 Intent。 

第 二 步 : 用 sentIntent 参数 为 传送 后 接受 的 广播 信息 PendingIntent。 

第 三 步 : 用 deliveryIntent 参数 为 送 达 后 接受 的 广播 信息 PendingIntent。 
第 四 步 : 发 送 SMS 短信 。 

第 五 步 ， 有 异常 则 用 mTextView01.setText(e.toString()) 输 出 异常 。 

引 体 代码 如 下 。 


访 创建 SmsManager 对 象 */ 
SmsManager smsManager = SmsManager.getDefault(); 


DODODD 


try 

{ 
上 创建 自 定义 Action 常数 的 Intent( 给 PendingIntent 参数 之 用 ) */ 
Intent itSend = new Intent(SMS SEND ACTIOIN); 
Intent itDeliver = new Intent(SMS_ DELIVERED _ ACTION); 


人 # sentIntent 参数 为 传送 后 接受 的 广播 信息 PendingIntent */ 
PendingIntent mSendPI = PendingIntent.getBroadcast 
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(getApplicationContext(), 0, itSend, 0); 


/* deliveryIntent 参数 为 送 达 后 接受 的 广播 信息 PendingIntent 
PendingIntent mDeliverPI = PendingIntent.getBroadcast 
(getApplicationContext(), 0, itDeliver, 0); 


谍 发 送 SMS 短信 ， 注 意 倒数 的 两 个 PendingIntent 参数 */ 
smsManager.sendTextMessage 
(strDestAddress, null, strMessage, mSendPl, mDeliverPD); 


mTextView01.setText(R.string.str_sms_sending); 


} 


catch(Exception e) 


{ 
mTextView01.setText(e.toString()); 


e.printStackTrace(); 


由 
} 


此 / 


5) 自 定 义 mServiceReceiver 对 象 ， 用 于 履 盖 BroadcastReceiver 监听 短信 状态 信息 。 如 


果 发 送 短信 成 功 则 输出 “成 功 发 送 ” 提 示 ， 如 果 发 送 短信 失败 则 输 昌 
体 代 码 如 下 。 


上 “发 送 失败 ”提示 。 基 


/六 自 定义 mServiceReceiver 履 盖 BroadcastReceiver 聆听 短信 状态 信息 */ 


public class mServiceReceiver extends BroadcastReceiver 
{ 

(QOverride 

public void onReceive(Context context, Intent intent) 


{ 


try 
{ 
/* android.content.BroadcastReceiver.getResultCode() 方 法 */ 
switch(getResultCode()) 
{ 
case Activity.RESULT OK: 
庆 发 送 短信 成 功 汉 
ImTextView01.setText(R.string.str_ sms_ sent success); 
mMakeTextToast 
( 
getResources().getText 
(R.string.str sms _ sent success).toString(), 
true 


298 国画 


第 6 章 手机 自动 服务 


break; 

case SmsManager.RESULT ERROR GENERIC FAILURE: 
庆 发 送 短信 失败 沁 
ImTIextView01.setText(R.string.str_ sms_sent failed); 
mMakeTextToast 
( 


getResources().getText 


(R.string.str_sms_ sent failed).toString(), 
true 
); 
break; 
case SmsManager.RESULT ERROR RADIO OFF: 
break; 
case SmsManager.RESULT ERROR NULL PDU: 
break; 


} 


catch(Exception e) 


{ 


mTextView01.setText(e.toString()); 
e.getStackTrace(); 


} 


} 


6) 定义 方法 mMakeTextToast(String str boolean isLong)， 用 于 输出 发 送 成 功 还 是 失败 的 
提示 。 具 体 代 码 如 下 。 


public void mMakeTextToast(String str, boolean isLong) 
{ 
iisLong 一 true) 
{ 
‘Toast.makeText(example12.this, str, Toast.LENGTH LONG).show(); 
} 
else 


{ 
Toast.makeText(example12.this, str, Toast.LENGTH SHORT).show(); 


】 
7) 定义 onResume() 方 法 ， 此 时 Activity 会 被 重建 。 有 具体 代码 如 下 。 


(QOverride 
protected void onResume() 


{ 
/* 自 定义 IntentFilter 为 SENT_SMS _ACTIOIN Receiver */ 
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IntentFilter mFilter01 ; 
mFilter01 = new IntentFilter(SMS SEND ACTIOIN); 
mReceiver01 = new mServiceReceiver(); 
registerReceiver(mReceiver01, mFilter01); 


/# 自 定义 IntentFilter 为 DELIVERED_SMS _ACTION Receiver */ 
mFilter01 = new IntentFilter(SMS DELIVERED ACTION); 
mReceiver02 = new mServiceReceiver(); 
registerReceiver(mReceiver02, mFilter01); 


super.onResume(); 


} 


8) 定义 onPause() 方 法 ， 此 时 Activity 会 被 和 暂停。 具体 代 码 如 下 。 


(QOverride 
protected void onPause() 


{ 


/# 取消 注册 自 定义 Receiver */ 


unregisterReceiver(mReceiver01); 


unregisterReceiver(mReceiver02); 


super.onPause(); 
} 
} 


发 送 短 信 后 能 够 显示 短信 是否 成 功 的 提示 ， 如 图 6-30 所 示 。 


shili12 


: 13475959369 


图 6-30 ”短信 提示 
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在 移动 手机 设备 中 ， 开 机 后 一 般 都 会 显示 一 个 特定 的 开机 界面 。 在 本 节 的 内 容 中 ， 将 通 
过 一 个 具体 实例 的 实现 ， 介 绍 设计 开机 显示 程序 的 具体 过 程 。 本 实例 的 源 代码 保存 在 “ 光 
盘 :daimax6\example13”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


6.13.1 实现 原理 

前 面 讲解 的 Activity、Service 和 Broadcast Receiver， 都 是 在 开机 之 后 运行 的 。 实 际 上 此 
时 开机 事件 会 发 送出 一 个 Android.intent.action.BOOT_ COMPLETED 广播 信息 。 只 要 接收 到 
此 Action， 就 能 在 Receiver 中 打开 自己 的 程序 。 
在 本 实例 中 ， 包 含 了 一 个 主 程序 Activity 和 一 个 Broadcast Receiver 类 ， 只 要 此 程序 运行 
一 次 ， 以 后 只 要 一 开机 就 会 运行 这 个 程序 。 


6.13.2 具体 实现 

本 实例 的 主 程序 文件 是 example13.java 和 StartupIntentReceiverjava， 下 面 开 始 讲解 其 
体 实现 流程 。 

1. 文件 example13.java 

文件 example13.java 的 功能 是 定义 主 程序 Activity， 本 程序 只 要 运行 一 次 ， 在 以 后 开机 
时 ， 就 会 自动 运行 ， 并 以 欢迎 的 TextView 文字 作为 提示 文本 。 有 具体 代码 如 下 。 


package irdc.example13; 


汕 


import irdc.example13.R; 

import android.app.Activity; 
import android.os.Bundle; 
import android.widget.TextView; 


public class example13 extends Activity 

{ 
族 本 程序 上 只 要 运行 一 次 ， 在 以 后 开机 时 ， 就 会 自动 运行 % 
private TextView mTextView01; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


人 * 为 了 快速 提示 ， 程 序 仅 以 欢迎 的 TextView 文字 作为 展示 */ 
mTextView01 = (TextView)findViewByld(R.id.myTextView!); 
mTextView01.setText(R.string.str welcome); 
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} 
} 


2. 文件 StartuplntentReceiver.java 

在 文件 StartupIntentReceiverjava 中 ， 添 加 了 一 个 继承 BroadcastReceiver 类 的 Hippo 
Startup-IntentReceiver 类 ， 在 其 内 部 履 盖 了 onReceive0 方 法 ， 此 方法 会 接收 来 自 系 统 的 广 
播 。 此 onReceive(0 方 法 的 唯一 任务 是 把 自己 唤醒 ， 所 以 在 传 入 Intent 的 参数 中 的 第 二 个 参数 
是 指定 Activity 的 类 ， 最 后 以 start-Activity0 方 法 打开 并 运行 程序 。 具 体 代码 如 下 。 


package irdc.example13; 


import android.content.BroadcastReceiver; 
import android.content.Context; 
import android.content. Intent; 


/* 捕捉 android.intent.action.BOOT COMPLETED 的 Receiver 类 */ 
public class StartupIntentReceiver extends BroadcastReceiver 


{ 
(QOverride 
public void onReceive(Context context, Intent intent) 


{ 


/* 当 收 到 Receiver 时 ， 指 定 打开 此 程序 (examplel3.class) */ 
Intent mBootIntent = new Intent(context, example13.class); 


/* 设置 mtent 打开 为 FLAG ACTIVITY NEW TASK */ 
mBootIntent.setFlags(Intent.FLAG ACTIVITY NEW _ TASK); 


/* 将 Intent 以 startActivity 传送 给 操作 系统 */ 
context.startActivity(mBootIntent); 


} 
} 


执行 后 将 会 显示 预先 设置 的 开机 欢迎 语 ， 如 图 6-31 所 示 。 


[二 | 园 古 则 便 4:32Awv 


example13 


显示 开机 程序 ! 


图 6-31 执行 效果 
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在 移动 手机 应 用 中 , 娱乐 和 多 媒体 是 一 个 重要 的 构成 模块 。 主要 包含 了 屏保 、 图 片 、MP3 
播放 、 影 片 播 放 和 拍照 等 应 用 。 在 本 节 的 内 容 中 ， 将 通过 儿 个 典型 实例 的 实现 ， 来 详细 介绍 
Android 中 娱乐 和 多 媒体 编程 的 基本 知识 。 


本 


在 当前 手机 应 用 中 ， 通 常 需要 获取 一 副 图 片 的 宽 和 高 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具 
体 实例 的 实现 , 介绍 获取 指定 图 片 宽 和 高 的 基本 流程 。 本 实例 的 源 代 码 保存 在 “光盘 :daimaxX 
examplel1”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


7.1.1 实现 原理 

在 本 实例 中 ， 通 过 一 个 ListView 对 象 并 设置 了 setOnCreateContextMenuListener 方法 ， 当 
用 户 单 击 一 个 选项 后 ， 能 够 弹出 多 个 ContextView 菜单 ， 并 以 onContextItemSelected 来 判断 用 
户 是 单 击 了 哪个 选项 ， 最 后 分 别 获 取 图 片 的 宽 和 高 。 

在 具体 实现 上 ， 通 过 Bitmap 对 象 的 BitmapFactory.decodeResource0) 方 法 来 获取 预先 设 定 
的 图 片 “123.png” 然后 再 通过 Bitmap 对 象 的 getHeightO 方 法 和 getWidth0) 方 法 来 获取 图 片 的 


7.1.2 具体 实现 

本 实例 的 主 程序 文件 是 examplel.java， 下 面 开始 讲解 其 具体 实现 代码 。 

1) 先 声 明 一 个 TextView 变量 mTextView01 和 一 个 ImageView 变量 mImageView01， 然 
后 分 别 声明 Context 菜单 的 选项 常数 CONTEXT_ITEM1、CONTEXT ITEM2 和 CONTEXT_ 
ITEM3。 具 体 代码 如 下 。 


package irdc.examplel; 


import irdc.examplel.R; 

import android.app.Activity; 

import android.graphics.Bitmap; 

import android.graphics.BitmapFactory; 
import android.os.Bundle; 

import android.view.ContextMenu; 
import android.view.Menu; 


import android.view.Menultem; 
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import android.view.View; 

import android.view.ContextMenu.ContextMenuInfo; 

import android.widget.ImageView; 

import android.widget.ListView; 

import android.widget.TextView; 

public class examplel extends Activity 

{ 
/# 声 明 一 个 TextView 变量 与 一 个 ImageView 变量 */ 
private TextView mTextView01; 


private ImageView mImageView01; 
从 声 明 Context 菜单 的 选项 常数 */ 
protected static final int CONTEXT ITEM1 = Menu.FIRST; 
protected static final int CONTEXT ITEM2 = Menu.FIRST+1; 
protected static final int CONTEXT ITEM3 = Menu.FIRST+2; 


2) 先 通过 findViewById 构造 器 来 创建 TextView 和 ImageView 对 象 ， 然 后 将 Drawable 
中 的 图 片 “baby.png” 放 入 自 定义 的 InageView 中 。 具 体 代码 如 下 。 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/* 通 过 findViewByld 构造 器 创建 TextView 与 ImageView 对 象 */ 
mTextView01 = (TextView)findViewByld(R.id.myTextView!); 
mlmageView01= (ImageView)findViewById(R.id.myImageView!1); 
/* 将 Drawable 中 的 图 片 baby.png 放 入 自 定义 的 ImageView 中 */ 
ImImageView01.setImageDrawable(getResources(). 
getDrawable(R.drawable.baby)); 


3) 先 设置 OnCreateContextMenuListener 给 TextView， 这 样 在 图 片上 可 以 使 用 ContextMenu， 
然后 履 盖 OnCreateContextMenu 方法 来 创建 ContextMenu 类 的 选项 。 具 体 代码 如 下 。 


/# 设 置 OnCreateContextMenuListener 给 TextView 
* 在 图 片上 可 以 使 用 ContextMenu*/ 
mImageView01.setOnCreateContextMenuListener 


(new ListView.OnCreateContext MenuListener() 

{ 
(QOverride 
/# 履 盖 OnCreateContextMenu 方法 来 创建 ContextMenu 类 的 选项 */ 
public void onCreateContext Menu 


(ContextMenu menu, View v, ContextMenuInfo menuInfo) 


{ 
menu.add(Menu.NONE, CONTEXT ITEM1, 0, R.string.str contextl]); 


menu.add(Menu.NONE, CONTEXT ITEM2, 0, R.string.str_context2); 
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menu.add(Menu.NONE, CONTEXT _ ITEM3, 0, R.string.str_context3); 
)); 
} 


4) 覆盖 OnContextItemSelected 方法 来 定义 用 户 单 击 菜单 后 的 动作 ， 然 后 通过 自 定 义 
Bitmap 对 象 BitmapFactory.decodeResource 来 取得 预 设 的 图 片 资 源 。 具 体 代码 如 下 。 


(QOverride 
/# 履 羡 OnContextItemSelected 方法 来 定义 用 户 单 击 菜 单 后 的 动作 */ 
public boolean onContextItemSelected(Menultem item) 


{ 


储 自 定义 Bitmap 对 象 并 通过 BitmapFactory.decodeResource 取得 
* 预 完 Impott 至 Drawable 的 baby.png 图 像 */ 
Bitmap myBmp = BitmapFactory.decodeResource 
(getResources(), R.drawable.baby); 
/# 通 过 Bitmap 对 象 的 getHight 与 getWidth 来 取得 图 片 宽 高 */ 
int intHeight = myBmp.getHeight(); 
int intWidth = myBmp.getWidth(); 


人 


4 


5) 根据 用 户 选 择 的 选项 ， 分 别 通 过 getHeight0 方 法 和 getWidth0) 方 法 获取 对 应 图 片 宽 度 
和 高 度 。 具 体 代码 如 下 。 


try 
{ 
放 菜 单 选项 与 动作 */ 
switch(item. getItemld()) 
{ 
人/# 将 图 片 宽度 显示 在 TextView 中 */ 
case CONTEXT _ ITEMI1: 
String strOpt = 
getResources(.getString(R.string.str_width) 
+"="+Integer.toString(int Width); 
mTextView01.setText(strOpt); 
break; 
人/# 将 图 片 高 度 显 示 在 TextView 中 */ 
case CONTEXT _ ITEM2: 
String strOpt2 = 
getResources().getString(R.string.str_ height) 
+"="+Integer.toString(intHeight); 
mTextView01.setText(strOpt2); 
break; 
人/# 将 图 片 宽 高 显示 在 TextView 中 */ 
case CONTEXT ITEM3: 
String strOpt3 = 
getResources().getString(R.string.str_width) 
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+"="+Integer.toString(int Width)+"\n" 
+getResources().getString(R.string.str height) 
+"="+Integer.toString(intHeight); 
mTextView01.setText(strOpt3); 
break; 


} 
} 
catch(Exception e) 
{ 
e.printStackTrace(); 
} 
return super.onContextltemSelected(item); 
} 
} 


执行 后 的 初始 效果 如 图 7-1 所 示 ， 当 长 时 间 选 中 图 片 后 会 弹出 用 户 选项 ， 如 图 7-2 所 示 。 


当选 择 一 个 选项 后 ， 会 弹出 对 应 的 获取 数值 。 
碟 关 全 6:21AM 


example1 
example1 


图 7-1 初始 效果 图 7-2 弹出 选项 
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当前 手机 应 用 中 ， 很 多 游戏 程序 中 经 常 需要 绘制 几何 图 形 。 在 本 节 的 内 容 中 ， 将 通过 一 
个 具体 实例 的 实现 , 介绍 在 Android 中 绘制 几何 图 形 的 实现 流程 。 本 实例 的 源 代码 保存 在 “ 光 
担 :\daima\7N\example2”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


7.2.1 实现 原理 
在 本 实例 中 ， 通 过 Android.graphics 的 类 来 绘制 2D 向 量 图 。 在 其 包 中 提供 了 很 多 在 手机 
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上 绘制 图 形 的 类 和 方法 。 例 如 ，Canvas 类 相当 于 一 个 图 纸 ， 所 有 的 图 形 都 在 它 上 面 绘制 并 显 


示 出 来 。 而 Paint 类 则 像 铅笔 ， 可 以 设置 为 不 同 的 颜色 ， 从 而 绘制 不 同 颜色 的 图 形 。 


7.2.2 具体 实现 


本 实例 的 主 程序 文件 是 example2.java， 下 面 开 始 讲解 其 具体 实现 代码 。 


1) 先 加 载 相关 类 ， 然 后 设置 ContentView 为 自 定义 的 MyView。 


L 体 代码 如 下 。 


public class example2 extends Activity 


{ 
(QOverride 
public void onCreate(B 


{ 


undle savedInstanceState) 


super.onCreate(savedInstanceState); 

/* 设置 ContentView 为 自 定义 的 MyView */ 
MyView myView = new MyView(this); 
setContentView(myView); 


} 


2) 自 定义 继承 View 类 的 MyView 对 象 ， 具 体 代 码 如 下 。 


/* 自 定义 继承 View 的 MyView */ 


private class MyView extends View 


{ 


public MyView(Context context) 


{ 
super(context); 


} 


3) 分 别 设置 背景 和 消除 锯齿， 然后 分 别 绘制 空心 国 


形 、 空 心 正 方形 、 


椭圆 形 、 空 心 三 角形 和 空心 梯形 。 具 体 代码 如 下 。 


/# 覆盖 onDraw0 方 法 */ 


(QOverride 


protected void onDraw(Canvas canvas) 


{ 


super.onDraw(canvas); 
庆 设置 背景 为 白色 * 


canvas.drawColor(Color.WHITE); 


Paint paint = new Paint() 


庆 去 锯齿 六 
paint.setAntiAlias(true); 


》 


/# 设置 paint 对 象 的 颜色 */ 
paint.setColor(Color.RED); 

设置 paint 对 象 的 style 为 STROKE: 空心 的 * 
paint.setStyle(Paint.Style.STROKE); 


罕 心 长 方形 、 空 心 
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庆 设置 paint 对 象 的 外 框 宽度 */ 
paint.setStrokeWidth(3); 


洱 


旋 画 一 个 空心 圆 形 */ 
canvas.drawCircle(40,40,30, paint); 
i 
canvas.drawRect(10,90,70,150,paint); 
VE 
canvas.drawRect(10,170,70,200,paint); 
主 画 一 个 空心 椭圆 形 */ 


RectF re=new RectF(10,220,70,250); 
canvas.drawOval(re, paint); 

le 八 全 Ns3% 儿 六 人 / 

Path path = new Path(); 
path.moveTo(10,330); 
path.lineTo(70,330); 
path.lineTo(40,270); 

path.close(); 


一 


canvas.drawPath(path, paint); 
族 画 一 个 空心 梯形 * 

Path pathl = new Path(); 
pathl.moveTo(10,410); 
pathl.lineTo(70,410); 
pathl.lineTo(55,350); 
pathl.lineTo(25,350); 
path1.close(); 
canvas.drawPath(path1, paint); 


4) 设置 实心 样式 和 颜色 ， 然 后 分 别 绘制 实心 
形 、 实 心 三 角形 和 实心 梯形 。 具 体 代 码 如 下 。 


庆 设置 paint 对 象 的 style 为 FILL: 实心 */ 
paint.setStyle(Paint. Style.FILL); 

上 庆 设置 paint 的 颜色 */ 
paint.setColor(Color.BLUE); 
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/* 


画 一 个 实心 圆 */ 


canvas.drawCircle(120, 40, 30, paint); 


Wt 


画 一 个 实心 正方 形 凤 


canvas.drawRect(90,90,150,150,paint); 


/* 


can 


vas.drawRect(90,170,150,200,paint); 


WE 


画 一 个 实心 椭圆 形 六 


RectF re2=new RectF(90,220,150,250); 
canvas.drawOval(re2, paint); 

庆 画 一 个 实心 三 角形 q 

Path path2 = new Path(); 


圆 形 、 实 心 正方 形 、 
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path2.moveTo(90,330); 
path2.lineTo(150,330); 
path2.lineTo(120,270); 
path2.close(); 
canvas.drawPath(path2, paint); 
入 画 一 个 实心 梯形 * 

Path path3 = new Path(); 
path3.moveTo(90,410); 
path3.lineTo(150,410); 
path3.lineTo(135,350); 
path3.lineTo(105,350); 
path3.close(); 
canvas.drawPath(path3, paint); 


5) 设置 渐变 样式 和 颜色 ， 然 后 分 别 绘制 渐变 圆 形 、 渐 变 正方 形 、 渐 变 长 方形 、 渐 变 椭 
形 、 渐 变 三 角形 和 渐变 梯形 。 具 体 代码 如 下 。 


启 设置 渐变 色 * 

Shader mShader=new LinearGradient(0, 0,100,100, 
new int[]{Color.RED, Color.GREEN ,Color.BLUE,Color.YELLOW;}, 
null, Shader.TileMode.REPEAT); 

paint.setShader(mShader); 


[ 河 [ 


族 画 一 个 渐变 色 的 圆 形 六 
canvas.drawCircle(200,40, 30, paint); 

A# 夯 一 个 渐变 色 的 正方 形 */ 
canvas.drawRect(170,90,230,150,paint); 
放 画 一 个 渐变 色 的 长 方形 */ 
canvas.drawRect(170,170,230,200,paint); 
入 画 一 个 渐变 色 的 椭圆 形 % 

RectF re3=new RectF(170,220,230,250); 
canvas.drawOval(re3, paint); 

让 画 一 个 渐变 色 的 三 角形 */ 

Path path4 = new Path(); 
path4.moveTo(170,330); 
path4.lineTo(230,330); 
path4.lineTo(200,270); 

path4.close(); 

canvas.drawPath(path4, paint); 

上 画 一 个 渐变 色 的 梯形 沁 

Path path$ = new Path(); 
path$.moveTo(170,410); 
path$.lineTo(230,410); 
path$.lineTo(215,350); 
path$.lineTo(185,350); 

path$.close(); 

canvas.drawPath(path5, paint); 
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6) 通过 canvas.drawText() 方 法 实现 写字 功能 ， 具 体 代码 如 下 。 


/# 写字 */ 
paint.setTextSize(24); 
canvas.drawText(getResources().getString(R.string.str_text1), 
240,50,paint); 
canvas.drawText(getResources().getString(R.string.str_text2), 
240,120,paint); 
canvas.drawText(getResources().getString(R.string.str text3)， 
240,190,paint); 
canvas.drawText(getResources().getString(R.string.str_text4), 
240,250,paint); 
canvas.drawText(getResources().getString(R.string.str text$), 
240,320,paint); 
canvas.drawText(getResources().getString(R.string.str text6), 
240,390,paint); 


执行 后 将 会 在 屏幕 内 显示 不 同 的 图 形 ， 并 在 后 面 显示 对 应 的 文字 描述 。 效 果 如 图 7-3 
所 示 。 


Vexample2 


|] Ba WA 长 王 江 


OOi@ a 


八 全 全 = 
八 展 尼 * 


图 7-3 执行 效果 
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在 手机 应 用 中 ， 屏 幕 保 护 程 序 是 一 个 十 分 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具 
体 实例 的 实现 ， 介 绍 在 Android 中 实现 屏保 功能 的 基本 流程 。 本 实例 的 源 代码 保存 在 “ 光 
盘 :\daimav7\example3”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


7.3.1 实现 原理 

本 实例 的 主要 难点 如 下 。 
口 控制 和 判断 用 户 静 止 未 触动 手机 键盘 或 屏幕 的 时 间 及 其 事件 。 
口 动态 全 屏幕 淡 入 、 淡 出 和 图 片 交 换 效果 。 
上 述 难 点 都 是 通过 线程 实现 的 ， 它 以 时 间 戳 记录 的 方式 ， 判 断 距离 上 一 次 单 击 键盘 或 屏 
幕 的 时 间 ， 并 计算 两 次 的 时 间 间 隔 。 当 超过 了 设置 了 时 间 后 ， 会 进行 屏保 ， 本 实例 设置 的 时 
间 间 陋 为 5 秒 。 


7.3.2 具体 实现 


本 实例 的 主 程序 文件 是 example3.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 先 加 载 相关 类 ， 然 后 设置 LayoutInflater 对 象 供 新 建 的 AlertDialog 使 用 。 具 体 代码 
如 下 。 


public class example3 extends Activity 


{ 


private TextView mTextView01; 
private ImageView mImageView0!1; 


/* LayoutInflater 对 象 作为 新 建 AlertDialog 之 用 */ 
private LayoutInflater mInflater0 1; 


) 定义 mView01 对 象 ， 用 于 输入 解锁 的 View。 通 过 Menu 选项 identifier， 用 以 识别 对 
4 事件 。 有 具体 代码 如 下 。 
/# 输入 解锁 的 View */ 


private View mView01; 
private EditText mEditText01,mEditText02; 


旋 Menu 选项 identifier， 用 以 识别 事件 */ 

static final private int MENU_ ABOUT = Menu.FIRST; 
static final private int MENU_EXIT = Menu.FIRST+!1; 
private Handler mHandler01 = new Handler(); 


private Handler mHandler02 = new Handler(); 
private Handler mHandler03 = new Handler(); 
private Handler mHandler04 = new Handler(); 


3) 分 别 定义 控制 User 静止 与 否 的 Counter， 控 制 Fade In 与 Fade Out 的 Counter， 控 制 循 
序 蔡 换 背 景 图 ID 的 Counter。 具 体 代 码 如 下 。 
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4) 


控制 User 静止 与 否 的 Counter */ 
private int intCounterl, intCounter2; 

/* 控制 Fade#In 与 Fade Out 的 Counter */ 
private int intCounter3, intCounter4; 

访 控制 循序 替换 背景 图 的 Counter */ 
private int intDrawable=0; 


设置 tmePeriod。 设 置 静止 超过 ns 将 自动 进入 屏 医保 护 。 具 体 代码 如 下 。 


族 上 一 次 用 户 有 动作 的 时 间 戳 */ 
private Date lastUpdateTime; 

旋 计算 用 户 多 少 秒 没有 动作 */ 

private long timePeriod; 

静止 超过 ns 将 自动 进入 屏幕 保护 */ 
private float fHoldStillSecond = (float) 5; 
private boolean bIfRunScreenSaver; 


private boolean bFadeFlagOut, bFadeFlagIn = false; 
private long intervalScreenSaver = 1000; 

private long intervalKeypadeSaver = 1000; 

private long intervalFade = 100; 

private int screen Width, screenHeight; 


ID 


5) 设置 每 隔 5s 置换 图 片 ， 用 Screen 保存 需要 用 到 的 背景 图 。 具 体 代 码 如 下 。 


6) 设置 在 setContentView 之 前 调用 全 屏幕 显示 ， 通 过 lastUpdateTime 方法 取得 | 
手机 的 时 间 ， 并 用 recoverOriginalLayout() 方 法 来 初始 化 Layout 上 的 Widget 的 可 见 性 。 有 具体 
代码 如 下 
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旋 每 n 秒 置 换 图 片 * 
private int intSecondsToChange = 5; 


入 设置 Screen 保存 需要 用 到 的 背景 图 */ 
private static int[] screenDrawable = new int[] 


{ 


R.drawable.screen1， 


R.drawable.screen2， 
R.drawable.screen3, 
R.drawable.screen4, 
R.drawable.screen5 


同 


o 


(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


/# 必须 在 setContentView 之 前 调用 全 屏幕 显示 */ 


} 


7) 设置 菜单 群 组 ID， 然 后 通过 menu.add 创建 
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requestWindowFeature(Window.FEATURE NO_TITLE); 
getWindow().setFlags 
( 
WindowManager.LayoutParams.FLAG FULLSCREEN, 
WindowManager.LayoutParams.FLAG FULLSCREEN 
); 


setContentView(R.layout.main); 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 
mlmageView01 = (ImageView)findViewById(R.id.myImageView1l); 
mbEditText01 = (EditText)findViewById(R.id.myEditTextl); 


旋 取得 用 户 触 碰 手机 的 时 间 3 
lastUpdateTime = new Date(System.currentTimeMillis()); 


/* 初始 化 Layout 上 的 Widget 可 见 性 */ 
recoverOriginalLayout(); 


体 代 码 如 下 。 


(QOverride 
public boolean onCreateOptionsMenu(Menu menu) 


{ 


} 


8) 根据 用 户 选择 的 菜单 项 ， 显 示 对 应 的 AlertDialog 提示 机 


/x* menu 群 组 ID */ 
int idGroup1l = 0; 


/* The order position of the item */ 
int orderMenuIteml = Menu.NONE; 
int orderMenuItem2 = Menu.NONET+1; 
/* 创建 具有 子 项 的 菜单 */ 
menu.add 
( 

idGroup1, MENU _ ABOUT, orderMenultem!l, R.string.app about 
); 
雍 创建 退出 菜单 * 


坏 有 子 项 的 沫 单 ， 最 后 创建 退出 染 单 。 具 


menu.add(idGroupl1, MENU_ EXIT, orderMenuItem2, R.string.str_exit); 


menu.setGroupCheckable(idGroup!1, true, true); 


return super.onCreateOptionsMenu(menu); 


[sal 
o 


具体 代码 如 下 。 
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(QOverride 
public boolean onOptionsItemSelected(Menultem item) 
{ 
switch(item.getltemld()) 
{ 
case (MENU ABOUT): 
new AlertDialog.Builder 
( 

example3.this 
).setTitle(R.string.app_about).setIcon 
( 

R.drawable.hippo 
).setMessage 
( 

R.string.app_about msg 
).SetPositiveButton(R.string.str_ok， 
new DialogInterface.OnClickListener() 
{ 

public void onClick 

(DialogInterface dialoginterface, int 1) 

{ 

} 

}).show); 
break; 

case (MENU EXIT): 
上 # 离开 程序 */ 


finish(); 
break; 
} 
return super.onOptionsItemSelected(item); 


} 


9) 
的 时 间 间 距 ， 如 果 静 止 不 动 ， 超 过 设置 的 5s， 则 运行 对 应 的 线程 。 具 体 代码 如 下 。 


谨 监控 User 没有 动作 的 运行 线程 */ 
private Runnable mTasks01 =new Runnable() 


{ 
public void run() 


{ 
intCounter1++; 
Date timeNow = new Date(System.currentTimeMillis()); 


/* 计算 用 户 静 止 不 动作 的 时 间 间 距 *%/ 


timePeriod = 


(long)timeNow.getTime() - (long)lastUpdateTime.getTime(); 
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用 mTasks01 监控 用 户 是 否 有 动作 的 运行 线程 ， 通 过 timePeriod 计算 用 户 前 


止 不 动作 


第 7 章 
float timePeriodSecond = ((float)timePeriod/1000); 
人 # 如 果 超 过 时 间 静 止 不 动 */ 


if(timePeriodSecond>fHoldStillSecond) 
{ 


入 静止 超过 时 间 第 一 次 的 标记 */ 
if(bIfRunScreenSaver—false) 


{ 


从 启动 运行 线程 2 *%/ 
mHandler02.postDelayed(mTasks02, intervalScreenSaver); 


/* Fade Out*/ 
if(intCounterl%(intSecondsToChange)==0) 
{ 
bFadeFlagOut=true; 
mHandler03.postDelayed(mTasks03, intervalFade); 
} 
else 
{ 
/* 在 Fade Out 后 立即 Fade Im */ 
if(bFadeFlagOut==true) 
{ 
bFadeFlagIn=true; 
mHandler04.postDelayed(mTasks04, intervalFade); 
} 
else 
{ 
bFadeFlagIn=false; 
intCounter4 = 0; 
mHandler04.removeCallbacks(mTasks04); 
} 
intCounter3 = 0; 
bFadeFlagOut = false; 
} 
blfRunScreenSaver = true; 
} 
else 


{ 


if(intCounterl%(intSecondsToChange)==0) 


{ 
bFadeFlagOut=true; 


mHandler03.postDelayed(mTasks03, intervalFade); 
} 


else 


{ 
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) 
所 


久 在 Fade Out( 淡 出 ) 后 立即 Fade mm《〈 淡 入 ) */ 
if(bFadeFlagOut==true) 
{ 
bFadeFlagIn=true; 
mHandler04.postDelayed(mTasks04, intervalFade); 
} 
else 
{ 
bFadeFlagIn=false; 
intCounter4 = 0; 
mHandler04.removeCallbacks(mTasks04); 
} 
intCounter3 = 0; 
bFadeFlagOut=false; 
} 


} 


else 

{ 
从 当 用 户 没有 动作 的 间距 未 超过 时 间 *%/ 
bIfRunScreenSaver = false; 
从 恢复 原来 的 Layout Visible*/ 
recoverOriginalLayout(); 


} 


/* 以 LogCat 监 看 User 静止 不 动 的 时 间 间 距 */ 
Log.i 
( 
"HIPPO", 
"Counterl:"+Integer.toString(intCounter1)+ 
/a 
Float.toString(timePeriodSecond)); 


和 # 反 复 运行 运行 线程 1% 
mHandler01.postDelayed(mTasks01, intervalKeypadeSaver); 


10) 定义 mTasks02， 设 置 每 1 秒 运 行 一 次 屏保 程序 ， 并 隐藏 原 有 Layout 上 面 的 Widget， 


调用 ScreenSaver() 方 法 加 载 图 片 ， 即 轮换 显示 预 设 的 $ 幅 图 片 。 


了 


/* Screen Saver Runnable */ 


private Runnable mTasks02 = new Runnable() 


{ 


public void run() 


{ 
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if(bIfRunScreenSaver==true) 


所 体 代码 如 下 。 
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intCounter2++; 


hideOriginalLayout(); 
showScreenSaver(); 


Log.i("HIPPO", "Counter2:"+Integer.toString(intCounter2)); 
mHandler02.postDelayed(mTasks02, intervalScreenSaver); 

} 

else 

{ 
mHandler02.removeCallbacks(mTasks02); 

} 

} 
上 


区 


11) 定义 mTasks03， 通 过 setAlpha 方法 设置 ImageView ( 
具体 代码 如 下 。 


/* Fade Out 特效 Runnable */ 
private Runnable mTasks03 =new Runnable() 


{ 
public void run() 


{ 
if(bIfRunScreenSaver==true && bFadeFlagOut=——true) 


{ 


intCounter3++; 


匀 设置 ImageView 的 透明 度 渐 瞳 下 去 */ 
mlmageView01.setAlpha(255-intCounter3*28); 


Log.i("HIPPO", "Fade out:"+Integer.toString(intCounter3)); 
mHandler03.postDelayed(mTasks03, intervalFade); 

} 

else 

{ 
mHandler03.removeCallbacks(mTasks03); 

} 

} 
}; 


像 组 件 ) 的 透明 度 渐 上 暗 下 去 。 


12) 定义 mTasks03， 通 过 setAlpha 方法 设置 ImageView 的 透明 度 渐 亮 起 来 。 


纪 体 代码 如 下 。 


/* Fade In 特效 Runnable */ 
private Runnable mTasks04 =new Runnable() 


{ 
public void run() 
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{ 


if(bIfRunScreenSaver==true && bFadeFlagIn==true) 
{ 


intCounter4++; 


/* 设置 ImageView 的 透明 度 渐 亮 起 来 */ 
mlmageView01.setAlpha(intCounter4*28); 


mHandler04.postDelayed(mTasks04, intervalFade); 
Log.i("HIPPO", "Fade In:"+Integer.toString(intCounter4)); 

} 

else 

{ 
mHandler04.removeCallbacks(mTasks04); 

} 

} 
上 


13) 先 定义 recoverOriginalLayout0 方 法 ， 用 于 恢复 原 有 的 Layout 可 视 性 ， 然 后 定义 
hideOriginalLayout() 方 法 ， 用 于 隐藏 原 有 应 用 程序 里 的 布局 配置 组 件 。 具 体 代码 如 下 。 


族 恢复 原 有 的 Layout 可 视 性 */ 

private void recoverOriginalLayout() 

{ 
mTextView01.setVisibility(View.VISIBLE); 
ImEditText01.setVisibility(View.VISIBLE); 
mlmageView01.setVisibility(View.GONE); 

} 


可 


作 隐藏 大 有 应 用 程序 里 的 布局 配置 组 件 光 

private void hideOriginalLayout() 

{ 
谨 欲 隐藏 的 Widget */ 
ImTextView01.setVisibility(View.INVISIBLE); 
mEditText01.setVisibility(View.INVISIBLE); 


/* 开始 Screen 保存 */ 
private void showScreenSaver() 


{ 


if(intDrawable>4) 
{ 


intDrawable = 0; 


DisplayMetrics dm=new DisplayMetrics(); 
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getWindowManager().getDefaultDisplay().getMetrics(dmy); 
ScreenWidth = dm.widthPixels; 
ScreenHeight = dm.heightPixels; 
Bitmap bmp=BitmapFactory.decodeResource(getResources(), 
screenDrawable[intDrawable]); 


了 


14) 通过 Matrix 设置 比例 ， 使 用 Matrix.postScale0) 方 法 设置 新 维度 ReSize， 通 过 resizedBitmap 
对 象 设置 图 文件 至 屏幕 分 辨 率 ， 新 建 Drawable 对 象 myNewBitmapDrawable 用 于 放大 图 文件 


至 全 屏幕 ， 通 过 setVisibility(View.VISIBLE) 使 InageView 可 见 。 有 具体 代码 如 下 。 


7 


/* Matrix 比例 */ 
float scaleWidth = ((float) screen Width) / bmp.get Width(); 
float scaleHeight = ((float) screenHeight) / bmp.getHeight() ; 


Matrix matrix = new Matrix(); 
/* 使 用 Matrix.postScale 设置 维度 ReSize */ 
matrix.postScale(scale Width, scaleHeight); 


/* 了 ReSize 图 文件 至 屏幕 分 辨 率 */ 
Bitmap resizedBitmap = Bitmap.createBitmap 


( 
bmp,0,0,bmp.getWidth(),bmp.getHeight(),matrix,true 
); 


入 新 建 Drawable 放大 图 文件 至 全 屏幕 */ 
BitmapDrawable myNewBitmapDrawable = 


new BitmapDrawable(resizedBitmap); 
mImageView01.setImageDrawable(myNewBitmapDrawable); 


/# 使 ImageView 可 见 驮 
mImageView01.setVisibility(View.VISIBLE); 


入 每 间隔 设置 秒 数 置 换 图 片 ID， 于 下 一 个 runnable2 才 会 生效 */ 
if(intCounter2%intSecondsToChange==0) 
{ 


intDrawable++; 


} 


1$) 定义 方法 onUserWakeUpEvent()， 实 现 解 锁 和 加 密 处 型 


1。 具 体 代 码 如 下 。 


public void onUserWakeUpEvent() 
{ 


if(bIfRunScreenSaver==true) 


{ 
try 
1 
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/六 LayoutInflater.from 取得 此 Activity 的 context */ 
mInfater01 = LayoutInflater.from(example3.this); 


访 创建 解锁 密码 使 用 View 的 Layout */ 
mView0l = mInflater01.inflate(R.layout.securescreen, null); 


上 对 话 框 中 唯一 的 EditText 等 待 输入 解锁 密码 */ 
mEditText02 = 
(EditText) mView01.findViewById(R.id.myEditText2); 


/* 创建 AlertDialog */ 

new AlertDialog.Builder(this) 
.SetView(mView01) 
.SetPositiveButton("OK", 

new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog, int whichButton) 
{ 
/* 比较 输入 的 密码 与 原 Activity 里 的 设置 是 否 相 符 */ 
if(mEditText01.getText().toString().equals 
(mEditText02.getText().toString())) 
{ 
雍 当 密 码 正 确 才 真 的 解锁 屏幕 保护 装置 * 
resetScreenSaverListener(); 
} 
} 
}).show(); 
} 
catch(Exception e) 
{ 
e.printStack Trace(); 
} 
} 
} 
16) 定义 updateUserActionTime0， 用 于 统计 用 户 单 击 键盘 或 屏幕 的 时 间 间 隔 。 有 具体 流 
程 如 下 。 
口 第 一 步 : 取得 单 击 按键 事件 时 的 系统 Time Millis。 


生生 
口 第 二 步 : 重新 计算 单 击 按键 距离 上 一 次 静止 的 时 间 间 距 。 
LL 体 代 码 如 下 。 


public void updateUserActionTime() 
{ 
旋 取得 单 击 按键 事件 时 的 系统 Time Millis (时 间 波 动 ) */ 


Date timeNow = new Date(System.currentTimeMillis()); 
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入 重新 计算 单 击 按键 距离 上 一 次 静止 的 时 间 间 距 六 


timePeriod = 


(long)timeNow.getTime() - (long)lastUpdateTime.getTime(); 
lastUpdateTime.setTime(timeNow.getTime()); 


} 


屏幕 。 流 程 如 下 。 


17) 定义 方法 resetScreenSaverListener()， 用 于 重新 设置 
口 第 一 步 : 删除 现 有 的 Runnable。 

第 二 步 : 取得 单 击 按键 事件 时 的 系统 Time Millis。 

第 三 步 : 重新 计算 单 击 按键 距离 上 一 次 静止 的 时 间 间 距 。 
第 四 步 : 通过 bIfRunScreenSaver 取消 屏保 。 

第 五 步 : 恢复 原来 的 Layout Visible。 

体 代 码 如 下 。 
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DODDO 


public void resetScreenSaverListener() 

{ 
访 删除 现 有 的 Runnable */ 
mHandler01.removeCallbacks(mTasks01); 
mHandler02.removeCallbacks(mTasks02); 


旋 取得 单 击 按键 事件 时 的 系统 Time Millis */ 
Date timeNow = new Date(System.currentTimeMillis()); 
旋 重新 计算 单 击 按键 距离 上 一 次 静止 的 时 间 间 距 六 


timePeriod = 


| 


(long)timeNow.getTime() - (long)lastUpdateTime.getTime(); 
lastUpdateTime.setTime(timeNow.getTime()); 


/#* 取消 屏幕 保护 */ 


bIfRunScreenSaver = false; 


/* 重 置 Runnablel 与 Runnablel 的 Counter */ 
intCounterl = 0; 


intCounter2 = 0; 


/* 恢复 原来 的 Layout， 变 为 可 见 状态 */ 


recoverOriginalLayout(); 


/* 重新 postDelayed() 新 的 Runnable */ 
mHandler01.postDelayed(mTasks01, intervalKeypadeSaver); 
} 


18) 定义 onKeyDown(int keyCode, KeyEvent event) 方 法 ， 用 于 监听 用 户 的 触摸 单 击 事件 。 
具体 代码 如 下 。 


(@Override 
public boolean onKeyDown(int keyCode, KeyEvent event) 
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{ 
if(bIfRunScreenSaver==true && keyCode!=4) 


旋 当 屏幕 保护 程序 正在 运行 中 ， 触 动 解除 屏幕 保护 程序 */ 
onUserWakeUpEvent(); 
} 
else 
{ 
入 更 新 用 户 未 触动 手机 的 时 间 戳 记 六 
updateUserActionTime(); 
} 
return super.onKeyDown(keyCode, event); 
) 
(QOverride 
public boolean onTouchEvent(MotionEvent event) 


{ 


if(bIfRunScreenSaver==true) 


* 当 屏幕 保护 程序 正在 运行 中 ， 触 动 解除 屏幕 保护 程序 */ 
onUserWakeUpEvent(); 
} 


else 

{ 
谍 更 新 用 户 未 触动 手机 的 时 间 戳 记 光 
updateUserActionTime(); 

} 


return super.onTouchEvent(event); 


} 


(@Override 
protected void onResume() 


{ 
// TODO Auto-generated method stub 


mHandler01.postDelayed(mTasks01, intervalKeypadeSaver); 
super.onResume(); 


} 


19) 定义 方法 onPause0 ， 用 于 删除 运行 中 的 运行 线程 mHandler01、mHandler02、 
mHandler03 和 mHandler01。 有 具体 代码 如 下 。 


(@Override 
protected void onPause() 


{ 


try 
{ 
和 * 删除 运行 中 的 运行 线程 % 
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mHandler01.removeCallbacks(mTasks01); 
mHandler02.removeCallbacks(mTasks02); 
mHandler03.removeCallbacks(mTasks03); 
mHandler04.removeCallbacks(mTasks04); 

} 

catch(Exception e) 

{ 
e.printStack Trace(); 

} 

super.onPause(); 

) 
} 


执行 后 如 果 超 过 5 秒 不 动 键盘 或 屏幕 ， 则 会 进入 屏保 状态 ， 如 图 7-4 所 示 。 可 以 设置 屏 
保密 码 ， 当 输入 正确 的 密码 后 才能 解除 屏保 。 如 图 7-5 所 示 。 


bse | 
图 7-4 ”执行 效果 图 7-5 解锁 密码 


在 上 述 代码 中 ， 声 明 的 4 个 Runnable 接口 对 象 是 整个 程序 的 重点 。 
口 mTasks01: 设置 每 1 秒 检查 一 次 timePeriod， 并 监视 是 否 超过 5 秒 未 触发 。 超 过 5 秒 
则 将 bIRunScreenSaver 这 个 flag 更 改 为 tue， 并 启动 mTasks02。 

口 mTasks02: 设置 每 1 秒 运 行 一 次 屏保 程序 ， 并 隐藏 原 有 Layout 上 面 的 Widget， 并 调 
用 ScreenSaverO 加 载 图 片 ， 即 轮换 显示 预 设 的 $ 幅 图 片 。 
口 mTasks03: 是 Fade Out 特效 使 用 的 Runable， 每 0.1 秒 运行 一 个 scale〈 刻 度 )。 
口 mTasks04: 是 Fade In 特效 使 用 的 Runable， 每 0.1 秒 运行 一 个 scale。 


De 可 ER 
在 触摸 屏 手机 中 ， 点 击 移动 照片 的 功能 十 分 常见 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实 
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例 的 实现 , 介绍 在 Android 中 实现 点 击 移动 照片 功能 的 基本 流程 。 本 实例 的 源 代 码 保存 在 “ 光 
盘 :\daimaexample4”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


7.4.1 ”实现 原理 


在 本 实例 中 ,设计 了 一 个 ImageView 对 象 ， 并 使 用 了 Drawable 中 的 照片 ， 在 程序 开始 运 
行 时 ， 将 照片 放 在 屏幕 中 央 。 通 过 onTouchEvent 事件 来 处 理 单 击 、 拖 动 和 放 开 等 事件 来 完成 
拖 动 图 片 的 功能 。 并 且 设 置 了 ImageView 的 onClickListener 让 用 户 在 单 击 图 片 的 同时 ， 恢 复 
到 图 片 的 初始 位 置 。 


7.4.2 具体 实现 


本 实例 的 主 程序 文件 是 example4.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 先 加 载 相 关 类 然后 分 别 声明 ImageView 变量 mImageView01， 声 明 相 关 变 量 存储 图 片 
宽 高 和 位 置 ， 声 明 存 储 屏 幕 的 分 辨 率 变 量 。 具 体 代 码 如 下 。 


| 


public class example4 extends Activity 

{ 
/声明 ImageView 变量 */ 
private ImageView mImageView0!1; 
放声 明 相关 变量 作为 存储 图 片 宽 高 ,位 置 使 用 */ 
private int intWidth, intHeight, intDefaultX, intDefaultY; 
private float mX, mY; 
放声 明 存 储 屏 幕 的 分 辨 紊 变 量 */ 


private int intScreenX, intScreenyY; 


2) 先 通 过 DisplayMetrics 取得 屏幕 对 象 ， 然 后 通过 intScreenX 对 象 和 intScreenY 对 象 取 
得 屏幕 像素 ， 最 后 分 别 设置 图 片 的 宽 高 。 具 体 代码 如 下 。 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


入 取得 屏幕 对 象 */ 
DisplayMetrics dm = new DisplayMetrics(); 
getWindowManager().getDefaultDisplay().get Metrics(dm); 


旋 取得 屏幕 像素 */ 
intScreenX = dm.widthPixels; 
intScreenY = dm.heightPixels; 


放 设置 图 片 的 宽 高 忆 
intWidth = 100; 
intHeight = 100; 
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3) 通过 findViewByld 构造 器 创建 InageView 对 象 ， 然 后 将 图 片 赋值 给 ImageView 来 呈 
现 ， 并 通过 RestoreButton0 方 法 初始 化 按钮 ， 设 置 位 置 居中 。 具 体 代 码 如 下 。 


/# 通 过 findViewById 构造 器 创建 ImageView 对 象 */ 
mlmageView01 =(ImageView) fmndViewById(R.id.myImageViewl); 
居 将 图 片 从 Drawable 赋值 给 ImageView 来 呈现 */ 
mlmageView01.setImageResource(R.drawable.baby); 


谍 初始 化 按钮 位 置 居中 */ 
RestoreButton(); 


4) 定义 setOnClickListenermew Button.OnClickListener())， 当 单 击 ImageView 时 ， 还 原初 
始 位 置 。 具 体 代码 如 下 。 


应 


外 击 ImageView， 还 原初 始 位 置 */ 
ImImageView01.setOnClickListenernew Button.OnClickListener() 
{ 
(@Override 
public void onClick(View v) 
{ 


RestoreButton(); 


下 
} 


5) 定义 onTouchEvent(MotionEvent evenb 履 盖 触 探 事件。 首先 取得 手指 触 控 屏 幕 的 位 置 ， 
然后 实现 触 控 事件 的 处 理 ， 即 实现 下 面 的 动作 处 理 。 
口 移动 位 置 。 
口 离开 屏幕 。 
呈 体 代码 如 下 。 


/* 材 盖 触 控 事 件 光 
(QOverride 


public boolean onTouchEvent(MotionEvent event) 


{ 


放 取 得 手指 触 控 屏 幕 的 位 置 */ 
float x = event.getX(); 
float y = event.getY(); 


try 

{ 
诺 触 控 事 件 的 处 理 *%/ 
switch (event.getAction()) 


{ 
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} 


/点 击 屏幕 光 
case MotionEvent.ACTION DOWN: 
picMove(x, y); 
break; 
放 移 动 位 置 */ 
case MotionEvent.ACTION MOVE: 
picMove(x, y); 
break; 
放 离 开 屏 幕 * 
case MotionEvent.ACTION_UP: 
picMove(x, y); 
break; 


} 


}catch(Exception e) 


{ 


e.printStack Trace(); 


} 


return true; 


6) 定义 picMove(float x, float y) 方 法 ， 用 于 实现 图 片 移动 。 


放 移 动 图 片 的 方法 */ 
private void picMove(float x, float y) 


{ 
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/默认 微调 图 片 与 指针 的 相对 位 置 */ 
mX=x-(intWidth/2); 
mY=y-(intHeight/2); 


饶 防 图 片 超过 屏幕 的 相关 处 理 刀 
旋 防 止 屏 幕 向 右 超过 屏幕 */ 
if((mX+intWidth)>intScreenX) 
{ 

mX = intScreenX-intWidth:; 
} 
庶 防 止 屏幕 向 左 超过 屏幕 */ 
else if(mX<0) 
{ 

mX =0; 
} 
庶 防 止 屏幕 向 下 超过 屏幕 */ 
else if ((mY+intHeight)>intScreenY) 
{ 

mY=intScreenY -intHeight; 
} 
旋 防 止 屏 幕 向 上 超过 屏幕 */ 
else if (mY<0) 


由 体 代码 如 下 。 
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/# 通 过 log( 导 航 对 象 ) 来 查看 图 片 位 置 */ 

Log.i("jay", Float.toString(mX)+","+Float.toString(mY)); 

/* 以 setLayoutParams 方法 ， 重 新 安排 Layout 上 的 位 置 *%/ 

mlmageView01.setLayoutParams 

( 
new AbsoluteLayout.LayoutParams 
(intWidth,intHeight,(int) mX,(inbmyY) 

); 

} 


7) 定义 RestoreButton(0) 方 法 ， 用 于 还 原 ImageView 位 置 的 事件 处 理 。 具 体 代码 如 下 。 


/* 还 原 ImageView 位 置 的 事件 处 理 */ 
public void RestoreButton() 
{ 
intDefaultX = ((intScreenX-intWidth)/2); 
intDefaultY = ((intScreenY -intHeight)/2); 
/*Toast 还 原 位 置 坐 标 */ 
mMakeTextToast 
( 
"(+ 
Integer.toString(intDefaultX)+ 
J 


Integer.toString(intDefaultY)+")",true 
); 


/* 使 用 setLayoutParams 方法 ， 重 新 安排 Layout 上 的 位 置 */ 
mlmageView01.setLayoutParams 
( 
new AbsoluteLayout.LayoutParams 
(intWidth,intHeight,intDefaultX,intDefaultY) 
); 
} 


~ 全 


8) 定义 mMakeTextToast(String str, boolean isLong) 方 法 ， 是 用 于 自 定义 发 出 信息 的 方法 。 


TT 


具体 代码 如 下 。 


旋 自 定义 发 出 信息 的 方法 */ 
public void mMakeTextToast(String str, boolean isLong) 
{ 

if(isLong==true) 


{ 
Toast.makeText(example4 .this, str, Toast.LENGTH LONG).show(); 
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} 
else 
{ 


Toast.makeText(example4 .this, str, Toast.LENGTH SHORT).show(); 


} 


} 


执行 后 效果 如 图 7-6 所 示 , 并 能 在 屏幕 中 通过 鼠标 点 击 来 移动 指 


所 示 。 


[= 
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图 7-6 执行 效果 
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在 手机 应 用 中 ， 为 了 扩大 手机 的 容量 ， 经 常会 使 用 


定 图 片 的 位 置 , 如 图 7-7 
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图 7-7 移动 图 片 


些 图 片 或 媒体 文件 。 在 本 区 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 
存储 卡 中 图 片 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\ 和 example5”， 下面 开始 讲解 


本 实例 的 具体 实现 流程 。 


7.5.1 ”实现 原理 


在 本 实例 中 ， 定 义 了 获取 文件 列表 方法 getSDO， 将 内 存 中 扫 


<String> 的 方式 存储 ， 


7.5.2 具体 实现 


本 实例 的 主 程序 文人 
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4 利 
图 片 加 载 到 Gallery Widget 中 。 


| 


存储 卡 。 在 存储 卡 中 ， 通 常会 保存 一 


介绍 在 Android 中 获取 


首 过 的 照片 以 File List 


定义 的 InageAdapter 来 初始 化 Gallery 对 象 ， 最 后 将 存储 卡 中 的 


F 是 example5.java， 下 面 开始 讲 解 其 具体 实现 代码 。 
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1) 先 加 载 相关 类 , 然后 新 增 ImageAdapter 并 赋值 给 Gallery 对 象 , 最 后 设置 几 个 itemclick 
事件 。 有 具体 代码 如 下 。 


Listener 


山中 


public class example5 extends Activity 


{ 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
Gallery g = (Gallery) fndViewById(R.id.mygallery); 


/* 新 增 ImageAdapter 并 赋值 给 Gallery 对 象 */ 
g.setAdapter(new ImageAdapter(this,getSD())); 


/* 设 置 儿 个 itemclickListener 事件 */ 
g.setOnItemClickListener(new OnltemClickListener() 


{ 


public void onItemClick(AdapterView<?> parent, 
View v, int position, long id) 


法 
) 


数据 , 并 将 获取 的 图 片 显示 在 ArrayList 


AT 


2) 定 义 List<String> getSDO 方 法 ,然后 访问 SD 1 
列表 中 。 具 体 代码 如 下 。 


private List<String> getSD() 
{ 
谨 设 定 目前 所 用 路 径 六 
List<String> itnew ArrayList<String>(); 
File 全 new File("/sdcard/"); 
File[] files=f.listFiles(); 


/* 将 所 有 数据 返回 到 ArrayList 中 */ 
for(int 1=0;i<files.length;i++) 
{ 
File file=files[i]; 
if(getImageFile(file.getPath())) 
it.add(file.getPath()); 


} 


return it; 
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3) 定义 getImageFile(String fName) 方 法 ， 获 取 存 储 -| 
定数 据 类 型 。 有 具体 代码 如 下 。 


private boolean getImageFile(String fName) 


{ 


boolean re; 


人 取得 扩展 名 光 
String end=fName.substring(fName.lastIndexOf(".")+1, 
fName.length()).toLowerCase(); 


族 根据 扩展 类 型 决定 打开 方式 */ 
if(end.equals("jpg")llend.equals("gif"')|lend.equals("png") 
llend.equals("jpeg")llend.equals("bmp")) 


re=true; 
} 


else 


{ 


re=false; 


} 


return re; 


} 


pa 


内 的 图 


片 文人 


4) 定义 继承 于 ImageAdapter 的 子 类 BaseAdapter， 流 程 如 下 。 


Parag 


口 第 一 步 : 声明 变量 。 
口 第 二 步 : 声明 ImageAdapter 构造 器 。 
口 第 三 步 : 获取 Gallery 信息 。 

有 人体 代码 如 下 。 


public class ImageAdapter extends BaseAdapter 


{ 
放声 明 变量 %/ 
int mGalleryltemBackground; 


private Context mContext; 
private List<String> lis; 


/*mageAdapter 构造 器 */ 
public ImageAdapter(Context c,List<String> li) 


{ 
mContext = ¢; 


lis=li; 


/* 使 用 文件 res/values/attrs.xml 中 的 <declare-styleable> 定 义 


* Gallery 属性 .*/ 


TypedArray a = obtainStyledAttributes(R.styleable.Gallery); 


/取得 Gallery 属性 的 Index id*/ 
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EF， 并 根据 扩展 类 型 决 
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mGalleryItemBackground = a.getResourceld( 


R.styleable.Gallery android galleryItemBackground, 0); 


广 让 对 象 的 styleable 属性 能 够 反复 使 用 */ 
arecycle(); 


} 


rT 


5) 定义 如 下 儿 个 需要 和 覆盖 的 方法 


口 getCount: 用 于 返回 图 片 数目 。 
口 getItem: 用 于 返回 位 置 。 
口 getView: 用 于 返回 View 对 象 。 
体 代码 如 下 。 


~ 


必定 义 要 歼 盖 的 方法 getCount, 返 回 图 片 数 目 */ 
public int getCount() 


{ 


return lis.size(); 


} 


放 定 义 要 履 盖 的 方法 getItem， 用 于 返回 位 置 */ 
public Object getItem(int position) 


{ 


return position; 


} 


| 


族 定义 要 履 盖 的 方法 getttemId， 用 于 返回 position */ 
public long getItemId(int position) 
{ 


return position; 


} 


颇 定 义 要 履 盖 的 方法 getView, 用 于 返回 View 对 象 */ 
public View getView(int position, View convertView, 


ViewGroup parent) 


上 产生 ImageView 对 象 */ 

ImageView i= new ImageView(mContext); 

上 诺 设 定 图 传 给 imageView 对 象 */ 

Bitmap bm = BitmapFactory.decodeFile(lis. 
get(position).toString()); 

i.setImageBitmap(bm); 

放 重 新 设 定 图 片 的 宽 高 */ 

i.setScaleType(ImageView.ScaleType.FIT XY); 

/# 重 新 设 定 Layout 的 宽 高 */ 

i.setLayoutParams(new Gallery.LayoutParams(136, 88)); 

访 设 定 Gallery 背景 图 */ 

i.setBackgroundResource(mGalleryItemBackground); 


| 
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/# 传 输 imageView 对 象 */ 
return 1; 
} 
} 
} 


执行 后 将 会 获取 存储 卡 内 的 图 片 信息 , 并 以 Gallery 的 样式 显示 出 来 ,具体 如 图 7-8 所 示 。 
鼠标 单 击 后 ， 能 够 滑动 显示 图 片 ， 如 图 7-9 所 示 。 
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图 7-9 滑动 显示 


接 下 来 详细 讲解 下 本 实例 的 测试 过 程 。 
(1) 创建 虚拟 SD 卡 

使 用 cmd 命令 进入 到 android 的 SDK 的 Tools 目录 下 ， 执行 mksdcard 创建 。 例 如 ， 作 者 
的 tools 目录 为 : 


E:\skyland\android-sdk-windows\tools> 
则 创建 命令 为 : 


E:\skyland\android-sdk-windows\tools>mksdcard 128M sdcard.img 


其 中 第 一 个 参数 为 要 创建 的 sdcard 容量 大 小 (容量 大 小 自己 决定 )， 第 二 个 参数 为 sdcard 
的 名 字 。 

(2) 启动 装 有 sdcard 的 Android 模拟 器 

可 以 不 用 cmd 命令 ， 而 直接 在 eclipse 中 启用 。 具 体 方 法 如 下 。 

第 一 步 : 用 鼠标 右键 单 击 实例 工程 ， 在 弹出 菜单 中 依次 选择 “Run As” 一 “Run 
Configurations”， 命令 ， 如 图 7-10 所 示 。 
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Run Confieurations 


图 7-10 命令 
第 二 步 : 在 弹出 的 对 话 框 中 “Target” 选 项 卡 ， 在 最 后 一 行 中 输入 如 下 字符 。 
-sdcard c:\sdcard.img 


如 图 7-11 所 示 。 


Run Confieurations. 


图 7-11 输入 -sdcard ci\sdcard.img 


(3) 添加 文件 到 创建 的 SD 卡 

经 过 以 上 两 步 操作 后 ， 模 拟 器 已 经 运行 了 ， 下 面 开 始 添加 文件 到 创建 的 SD 卡 。 具 体 有 
如 下 两 种 方法 。 

第 一 种 : eclipse 菜单 实现 。 

在 eclipse 中 打开 DDMS 视图 ， 打 开 “File Explor” 选 项 卡 ， 单 击 目录 下 的 “sdcard”， 如 
7-12 所 示 。 


drwxrwx-—x 


.10 dry 


1. jpg 39863 2009-07-25 03:41 ----rwxr-x 

3. jpe 0 2010-03-21 07:42 ----rwxr-x 

8. jpg 23867 2010-03-21 07:31 ----rwxr-x 

EE DCIN 2010-03-21 07:33 d---rwxr-x 

久 LST.I 2010-03-21 07:28 d---rwxr-x 
BE systen 2009-11-24 22:02 drwxr-xr-x | 


图 7-12 “File Explor” 选 项 卡 


通过 单 击 图 7-12 中 的 围 | 按钮 ， 可 以 上 传 本 地 文件 到 虚拟 存储 卡 。 
第 二 种 : 用 cmd 命令 。 
通过 cmd 来 到 android 的 SDK 的 Tools 目录 下 ， 然 后 用 adb push 命令 添加 ， 格 式 如 下 。 


E:\skyland\android-sdk-windows\tools>adb push new.JPG /sdcard 


| | ei 
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其 中 ， 第 一 个 参数 为 要 加 入 的 图 片 的 全 名 ， 如 果 名 字 中 间 有 空格 ， 要 用 双 引 号 将 其 括 起 


。 例 如 ， 


adb push "c:\1.jpg" /sdcard 


第 二 参数 就 是 刚才 创建 的 sdcard 了 。 


Oe el lt 


在 移动 手机 中 ， 内 置 了 多 媒体 这 个 Intent， 用 户 可 以 直接 单 击 多 媒体 按钮 进入 菜单 程序 。 


当然 也 可 以 通过 编程 的 方式 实现 上 述 操作 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 


介 


获取 内 置 媒 体 中 图 片 文件 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\7\example6”， 


下 面 开 始 讲解 本 实例 的 具体 实现 流程。 


7.6. 


1 实现 原理 


睛 5 


7.6 


而 


在 本 实例 中 ， 当 单 击 屏幕 中 的 图 片 按钮 后 会 打开 图 片 集 画 面 ， 让 用 户 挑选 自己 喜欢 的 
主要 包括 如 下 工作。 

口 打开 mntentACTION GET_ CONTENT。 

口 获取 用 户 选择 的 Image。 

口 通过 ContentResolver 将 Image 转换 为 Bitmap。 

口 在 ImageView 中 显示 图 片 。 


.2 具体 实现 


本 实例 的 主 程序 文件 是 example6.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 先 加 载 相关 类 ， 然 后 在 onCreate 方法 中 引入 主 布局 文件 main.xml， 最 后 设置 myButton01 


对 象 ， 用 于 激活 单 击 事件 OnClickListener()。 具 体 代码 如 下 。 
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public class example6 extends Activity 
{ 

private Button myButton01; 

private ImageView myImageView01; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mylmageView01 = (ImageView) findViewBylId(R.id.myImageView01); 
myButton01 = (Button) fmndViewById(R.id.myButton01); 


2) 定义 myButton01 的 单 击 事件 OnClickListener0， 流 程 如 下 。 
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口 第 一 步 : 打开 Pictures 画面 类 型 设置 为 image。 

口 第 二 步 : 使 用 Intent 中 的 ACTION_GET_CONTENT 动作 。 
口 第 三 步 : 获取 照片 后 返回 原来 画面 。 

所 体 代 码 如 下 。 


myButton01.setOnClickListener(new Button.OnClickListener() 
{ 
(@Override 
public void onClick(View arg0) 
{ 
Intent intent = new Intent(); 
上/# 打开 Pictures 画面 类 型 设置 为 image */ 
intent.setType("image/*"); 
/# 使 用 Intent.ACTION_GET_CONTENT 这 个 Action */ 
intent.setAction(Intent.ACTION_ GET_ CONTENT); 
雍 取得 照片 后 返回 本 画面 */ 
startActivityForResult(intent, 1); 


DD 
} 


3) 定义 onActivityResult(int requestCode, int resultCode)， 用 于 处 理 用 户 挑选 的 图 片 。 通 过 
requestCode 和 resultCode 返回 标识 码 以 及 数据 类 型 为 Intent 的 数据 参数 ， 调 用 Intent 对 象 的 
getData(0) 方 法 可 以 获取 具体 内 容 。 有 具体 代码 如 下 。 


(@Override 
protected void onActivityResult(int requestCode, int resultCode, 
Intent data) 
{ 
if (resultCode == RESULT OK) 
{ 


Uri uri = data.getData(); 
ContentResolver cr = this.getContentResolver(); 
try 
{ 
Bitmap bitmap = BitmapFactory.decodeStream(cr 
.openInputStream(uri)); 
/* 将 Bitmap 设置 到 ImageView */ 
mylmageView01.setImageBitmap(bitmap); 
} catch (FileNotFoundException e) 
{ 
e.printStack Trace(); 
} 
} 
super.onActivityResult(requestCode, resultCode, data); 


} 
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) 


执行 后 的 初始 效果 如 图 7-13 所 示 。 单 击 “ 图 片 ”按钮 后 会 显示 系统 内 的 图 片 信息 ， 如 
图 7-14 所 示 。 选 中 某 幅 图 片 并 单 击 后 ， 会 突 


》 会 突出 显示 此 照片 a 


l 


图 片 


No media found. 


图 7-13 初始 效果 


图 7-14 系统 内 的 图 片 


注意 : 如 果 手 机 多 媒体 中 没有 图 片 ， 则 可 以 按照 7.5 中 的 操作 在 SD 存储 卡 中 添加 照片 ， 
同样 可 以 显示 出 对 应 效果 。 


7 


在 移动 手机 中 ， 调 节 音 量 大 小 是 一 个 十 分 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具 
体 实 例 的 实现 , 介绍 在 Android 手机 中 调节 音量 大 小 的 基本 流程 。 本 实例 的 源 代码 保存 在 “ 光 
描 :\daima\7\example7”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


7.7.1 ”实现 原理 


在 Android API 中 的 AudioManager 类 提供 了 相关 的 方法 ， 可 以 在 程序 中 控制 手机 音量 的 
大 小 ， 也 可 以 切换 声音 的 模式 为 震动 或 静音 


百 。 


本 实例 使 用 的 素材 图 片 存放 在 了 “res\drawable” 目 录 下 。 


7.7.2 具体 实现 


本 实例 的 主 程序 文件 是 example7.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 3 


加载 相 关 类 ， 然 后 声明 各 个 需要 的 变量 。 有 具体 代码 如 下 。 


RH 


public class example7 extends Activity 
人/# 变量 声明 汶 
private ImageView myImage; 
private ImageButton downButton， 


private ImageButton upButton; 
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private ImageButton normalButton; 
private ImageButton muteButton; 
private ImageButton vibrateButton; 
private ProgressBar myProgress; 
private AudioManager audioMa; 
private int volume=0; 


2) 依次 对 象 初始 化 变量 audioMa、myImage、myProgress、downButton、upButton、normal 
Button、muteButton、muteButton 和 vibrateButton 。 上 有 具体 代码 如 下 。 


(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


雍 对 象 初始 化 * 

audioMa = (AudioManager)getSystemService(Context.AUDIO SERVICE); 
myImage = (ImageView)findViewBylId(R.id.myImage); 

myProgress = (ProgressBar)findViewByld(R.id.myProgress); 

downButton = (ImageButton)findViewById(R.id.downButton); 

upButton = (ImageButton)findViewByld(R.id.upButton); 

normalButton = (ImageButton)findViewByld(R.id.normalButton); 
muteButton = (ImageButton)findViewByld(R.id.muteButton); 
vibrateButton = (ImageButton)findViewById(R.id.vibrateButton); 


3) 分 别 设置 初始 的 手机 音量 和 初始 的 声音 模式 ， 有 具体 代码 如 下 。 


久 设置 初始 的 手机 音量 汐 
volume=audioMa.getStreamVolume(AudioManager.STREAM _ RING); 
myProgress.setProgress(volume); 

旋 设置 初始 的 声音 模式 * 

int mode=audioMa.getRingerMode(); 
if(mode==AudioManager.RINGER MODE NORMAL) 

{ 


mylImage.setImageDrawable(getResources() 


.getDrawable(R.drawable.normal)); 


} 

else if(mode=—=AudioManager.RINGER MODE SILENT) 

{ 

mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.mute)); 

} 

else if(mode=——AudioManager.RINGER MODE VIBRATE) 

{ 


mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.vibrate)); 
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} 


4) 设置 音量 调 小 按钮 downButton 的 处 理事 件 setOnClickListener， 流 程 如 下 。 
口 第 一 步 : 设置 音量 调 小 一 格 。 
口 第 二 步 : 设置 调整 后 声音 模式 。 


L 体 代码 如 下 。 


入 音量 调 小 声 的 按钮 光 
downButton.setOnClickListener(new Button.OnClickListener() 
{ 
OOverride 
public void onClick(View arg0) 
{ 
旋 设置 音量 调 小 一 格 */ 
audioMa.adjustVolume(AudioManager.ADJUST LOWER, 0); 
volume=audioMa.getStreamVolume(AudioManager.STREAM RING); 
myProgress.setProgress(volume); 
庆 设置 调整 后 声音 模式 */ 
int mode=audioMa.getRingerMode(); 
if(mode==AudioManager.RINGER MODE NORMAL) 
{ 


mylImage.setImageDrawable(getResources() 


.getDrawable(R.drawable.normal)); 
} 
else if(mode==AudioManager.RINGER MODE SILENT) 
{ 
mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.mute)); 
} 
else if(mode==AudioManager.RINGER MODE VIBRATE) 
{ 
mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.vibrate)); 


} 
人 


5) 设置 音量 调 大 按钮 upButton 的 处 理 
口 第 步 : 设置 音量 调 大 一 格 。 
口 第 二 步 : 设置 调整 后 声音 模式 。 


L 体 代码 如 下 。 


入 音量 调 大 声 的 按钮 光 
upButton.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 


小 | 


有 件 setOnClickListener， 流 程 如 下 。 
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public void onClick(View arg0) 
{ 
庆 设置 音量 调 大 一 格 */ 
audioMa.adjustVolume(AudioManager.ADJUST RAISE., 0); 
volume=audioMa.getStreamVolume(AudioManager.STREAM RING); 
myProgress.setProgress(volume); 
旋 设置 调整 后 的 声音 模式 */ 
int mode=audioMa.getRingerMode(); 
if(mode==AudioManager.RINGER MODE NORMAL) 
{ 


mylImage.setImageDrawable(getResources() 


.getDrawable(R.drawable.normal)); 
} 
else if(mode==AudioManager.RINGER MODE SILENT) 
{ 
mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.mute)); 
} 
else if(mode==AudioManager.RINGER MODE VIBRATE) 
{ 
mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.vibrate)); 


} 


6) 设置 调整 正常 铃声 模式 按钮 normalButton 的 处 理事 件 setOnClickListener， 流 程 如 下 。 
口 第 一 步 : 设置 铃声 模式 为 NORMAL。 
口 第 二 步 : 设置 音量 与 声音 模式 。 


由 体 代码 如 下 。 


入 调整 铃声 模式 为 正常 模式 的 按钮 */ 
normalButton.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
上 # 设置 铃声 模式 为 正常 模式 */ 
audioMa.setRingerMode(AudioManager.RINGER MODE NORMAL); 
上 # 设置 音量 与 声音 模式 */ 
volume=audioMa.getStreamVolume(AudioManager.STREAM RING); 
myProgress.setProgress(volume); 


mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.normal)); 


1 
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7) 设置 调整 静音 铃声 模式 按钮 muteButton 的 处 理 
口 第 设置 铃声 模式 为 SILENT (静音 )。 


F setOnClickListener， 流 程 如 下 。 


| 
山 喇 
Mur 
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口 第 二 步 : 设置 音量 与 声音 状态 。 
所 体 代码 如 下 


族 调整 铃声 模式 为 静音 模式 的 按钮 */ 
muteButton.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
族 设置 铃声 模式 为 SILENT */ 
audioMa. et 
/#* 设置 音量 与 声音 状态 * 
0 eT 


myProgress.setProgress(volume); 


mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.mute)); 
} 
); 


有 件 setOnClickListener， 流 程 如 下 。 


8) 设置 调整 震动 铃声 模式 按钮 vibrateButton 的 处 到 
口 第 一 步 : 设置 铃声 模式 为 VIBRATE (震动 )。 

口 第 二 步 : 设置 音量 与 声音 状态 。 

体 代码 如 下 。 


族 调整 铃声 模式 为 震动 模式 的 按钮 */ 
vibrateButton.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
族 设置 铃声 模式 为 VIBRATE * 
audioMa. ts 
族 设置 音量 与 声音 状态 
人 


myProgress.setProgress(volume); 


| 


下 


mylImage.setImageDrawable(getResources() 
.getDrawable(R.drawable.vibrate)); 


执行 后 将 会 显示 一 个 音量 调节 界面 。 有 具体 如 图 7-15 所 示 。 
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图 7-15 执行 效果 


7.8 D0 MP30D0 


在 移动 手机 中 ， 播 放 MP3 文件 是 一 个 十 分 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具 
体 实例 的 实现 , 介绍 在 Android 手机 中 播放 MP3 文件 的 基本 流程 。 本 实例 的 源 代 码 保 存在 “ 光 
盘 :\daima\7vexample8”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


7.8.1 实现 原理 

在 本 实例 中 ， 插 入 了 3 个 按钮 ， 分 别 用 于 播放 、 和 暂停 和 停止 。 当 单 击 “ 播 放 ” 按 钮 后 
会 从 指定 的 手机 资源 中 获取 MP3 文件 ， 并 执行 播放 处 理 。 在 具体 实现 上 ， 先 添加 一 个 
MediaPlayer 对 象 ， 并 使 用 MediaPlayercreate0 方 法 来 创建 播放 器 资源 ， 然 后 通过 方法 
MediaPlay.create0、MediaPlaystop0 和 MediaPlaypause0 分 别 实 现 开 始 、 停 止 和 暂停 功能 。 
为 了 处 理 按钮 所 需要 的 各 个 事件 ,需要 窗 盖 各 个 ImageButton 的 onClick0) 方 法 , 通过 按钮 来 
控制 MediaPlayer 的 状态 。 


7.8.2 具体 实现 


本 实例 的 主 程序 文件 是 example8.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
1) 先 引入 相关 类 ， 然 后 分 别 声明 一 个 ImageButton、TextView 和 MediaPlayer 变量 ， 声 明 
一 个 Flag 作为 确认 音乐 是 否 暂 停 的 变量 并 默认 为 false。 有 具体 代码 如 下 。 


public class example8 extends Activity 

{ 
放声 明 一 个 ImageButton,TextView,MediaPlayer 变量 */ 
private ImageButton mButton01, mButton02, mButton03; 
private TextView mTextView01; 
private MediaPlayer mMediaPlayer01; 
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对 象 ， 然 后 创建 MediaPlayer 对 象 ， 并 将 音乐 以 Import 的 方式 存储 为 res/raw/always.mp3。 具 
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2) 


放声 明 一 个 Flag 作为 确认 音乐 是 否 暂 停 的 变量 并 默认 为 false*/ 


private boolean bIsPaused = false; 


先 引 入 主 布局 文件 main.xml， 通 过 findViewByld 构造 器 创建 TextView 与 ImageView 


体 代码 如 下 。 
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3) 
口 
口 


踢 放 。 


口 


public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 通 过 findViewById 构造 器 创建 TextView 与 InageView 对 象 */ 
mButton01 =(ImageButton) findViewByld(R.id.myButton!1); 
mButton02 =(ImageButton) findViewById(R.id.myButton2); 
mButton03 =(ImageButton) fndViewById(R.id.myButton3); 
ImTextView01 = (TextView) findViewById(R.id.myTextViewl); 


/* onCreate 时 创建 MediaPlayer 对 象 */ 

ImMediaPlayer01 = new MediaPlayer(); 

上 将 音乐 以 Impott 的 方式 存储 在 res/raw/always.mp3 */ 
mMediaPlayer01 = MediaPlayer.create(example8.this, R.raw.big); 


运行 播放 音乐 的 按钮 ， 流 程 如 下 。 
第 一 步 : 覆盖 OnClick 事件 。 
第 二 步 : 在 MediaPlayer 对 象 中 取得 播放 资源 并 使 用 停止 播放 方法 stop0 之 后 ， 开 始 台 


bnll 
[ny 


第 三 步 : 改变 TextView 为 开始 播放 状态 。 


L 体 代码 如 下 。 


雍 运行 播放 音乐 的 按钮 * 
mButton01.setOnClickListener(new ImageButton.OnClickListener() 
{ 

(QOverride 

/# 覆 盖 OnClick 事件 */ 

public void onClick(View V) 


{ 
try 
{ 
if (mMediaPlayer01 != null) 
{ 
mMediaPlayer01.stop(); 
} 


#* 在 MediaPlayer 取得 播放 资源 与 stop0 之 后 
* 要 准备 Playback 的 状态 前 一 定 要 使 用 MediaPlayer.prepare()*/ 
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mMediaPlayer01.prepare(); 
此 开始 或 恢复 播放 */ 
mMediaPlayer01.start(); 
人 # 改 变 TextView 为 开始 播放 状态 */ 
mTextView01.setText(R.string. str_start); 

} 

catch (Exception e) 

{ 
mTextView01.setText(e.toString()); 
e.printStackTrace(); 

} 

} 
D); 


4) 定义 停止 播放 处 理事 件 ， 通 过 mMediaPlayer01.stop0 停 止 播 放 MP3， 改 变 TextView 
对 象 为 停止 播放 状态 。 有 具体 代码 如 下 。 


镀 停止 播放 */ 
mButton02.setOnClickListener(new ImageButton.OnClickListener() 
{ 
OOverride 
public void onClick(View arg0) 
{ 
try 
{ 
if (mMediaPlayer01 != null) 
{ 
fee) 
mMediaPlayer01.stop(); 
放 改 变 TextView 为 停止 播放 状态 */ 
ImTextView01.setText(R.string.str_ close); 


} 


UT 


} 


catch (Exception e) 

{ 
/IODO Auto-generated catch block 
ImTextView01.setText(e.toString(O); 
e.printStackTrace(); 

} 

} 
); 


5) 定义 暂停 播放 处 理 习 


人 人 
mButton03.setOnClickListener(new ImageButton.OnClickListener() 


FE， 首先 判断 是 否 处 于 暂停 状态 。 具 体 代 码 如 下 。 


pn 
让 
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{ 
(QOverride 
public void onClick(View arg0) 
{ 
// TODO Auto-generated method stub 
try 
{ 
if (mMediaPlayer01 != nul) 
{ 
* 是 否 为 暂停 状态 = 人 否 */ 
if(bIsPaused==false) 
{ 
族 暂 停 播 放 %/ 
mMediaPlayer01.pause(); 
/设置 Flag 为 treu 表示 播放 状态 为 暂停 */ 
blIsPaused = true; 
/# 改 变 TextView 为 暂停 播放 */ 
mTextView01.setText(R.string.str pause); 
} 
必 是 否 为 暂停 状态 = 是 
else es 
{ 
从 恢复 播 出 状态 光 
mMediaPlayer01 .start(); 
/* 设 置 Flag 为 false 表示 播放 状态 为 非 暂 停 状态 */ 
bIsPaused = false; 
人 # 改 变 TextView 为 开始 播放 */ 
mTextView01.setText(R.string.str start); 


} 


catch (Exception e) 

{ 
/IODO Auto-generated catch block 
mTextView01.setText(e.toString()); 
e.printStackTrace(); 

} 

} 
); 


6) 通过 MediaPlayerOnCompletionLister 监听 是 否 播 放 完 毕 ， 并 和 宪 盖 文件 播 出 完 


并 改变 TextView 为 播放 结束 。 具 体 代码 如 下 。 


mMediaPlayer01.setOnCompletionListener( 
new MediaPlayer.OnCompletionListener() 


{ 


记 窗 盖 文件 播 出 完毕 事件 */ 
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public void onCompletion(MediaPlayer arg0) 
{ 
try 
{ 
/解除 资源 与 MediaPlayer 的 赋值 关系 
* 让 资源 可 以 为 其 他 程序 利用 */ 
mMediaPlayer01.release(); 
/# 改 变 TextView 为 播放 结束 */ 
mTlextView01.setText(R.string.str OnCompletionListener); 
} 
catch (Exception e) 
{ 
mTextView01.setText(e.toString()); 
e.printStackTrace(); 
} 
} 
D); 


7) 使 用 MediaPlayerOnErrorListener 监听 履 盖 错误 处 理事 件 ， 当 发 后 错误 时 也 解除 资源 
与 MediaPlayer 的 赋值 。 有 具体 代码 如 下 。 


/# 当 MediaPlayer.OnErrorListener 会 运行 的 Listener */ 
mMediaPlayer01.setOnErrorListener(new MediaPlayer.OnErrorListener() 
{ 
(QOverride 
访 履 盖 错 误 处 理事 件 */ 
public boolean onError(MediaPlayer arg0, int argl, int arg2) 
{ 
try 
{ 
记 发 生 错误 时 也 解除 资源 与 MediaPlayer 的 赋值 */ 
mMediaPlayer01 .release(); 
mTextView01.setText(R.string.str OnErrorListener); 
} 
catch (Exception e) 
{ 
mTextView01.setText(e.toString()); 
e.printStackTrace(); 


} 


return false; 


} 
8) 覆盖 主 程序 暂停 状态 事件 ， 在 主 程序 暂停 时 解除 资源 与 媒体 播放 器 的 赋值 关系 ，i 
catch 实现 异常 处 理 。 具 体 代 码 如 下 


Ed 
关 
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/* 窗 盖 主 程序 暂停 状态 事件 */ 


protected void onPause() 
{ 

try 

{ 


/在 主 程序 暂停 时 解除 资源 与 MediaPlayer 的 赋值 关系 */ 
ImMediaPlayer01.release(); 


} 


catch (Exception e) 


{ 


mTextViewO1.setText(e.toString()); 


e.printStack Trace(); 


} 


super.onPause(); 


鼻 放 按钮 播放 指定 的 MP3 文件 。 如 图 7-16 所 示 。 


[=] 辐 动 剧 拓 7:26w 


example8 


图 7-16 ”执行 效果 


注意 : 如 果 手 机 多 媒体 中 没有 MP3 文件 片 ， 则 可 以 按照 7.5 中 的 操作 在 SD 存储 卡 中 添 
加 MP3 资源 ， 同 样 可 以 显示 出 对 应 效果 。 


7.9 则 由 四 


在 移动 手机 中 ， 录 音 处 理 


也 是 


个 十 分 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 


实例 的 实现 ， 介 绍 在 Android 手机 中 实现 录音 处 理 的 基本 流程 。 本 实例 的 源 代 码 保存 在 “ 光 


盘 :daimax7\example9”， 下 面 居 


[ 始 计 


7.9.1 实现 原理 


F 解 本 实例 的 具体 实现 流程 。 


在 本 实例 中 ， 插 入 了 4 个 按钮 ， 分 别 用 于 录音 、 停 止 录音 、 播 放 录 音 和 删除 录音 。 为 了 


能 够 不 限制 录音 的 长 度 ， 现 将 录音 暂时 保存 到 存储 卡 ， 当 录音 完毕 后 ， 再 将 录音 文件 显示 在 
ListView。 单 击 文件 后 ， 可 以 播放 或 删除 录音 文件 。 
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具体 实现 


本 实例 的 主 程序 文件 是 example9.java， 下 面 开 始 讲解 其 具体 实现 代码 。 


1)〉 先 加 载 相关 类 ， 


2 


3) 通过 sdCardExit 对 象 判 断 SD Card 是 否 插入 ， 然 后 获取 SD Card 路 径 作 为 录音 的 文 从 


\ 记 


> 


位 置 ， 
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然后 分 别 声明 各 个 需要 的 变量 。 有 具体 代码 如 下 。 


public class example9 extends Activity 


{ 


private ImageButton myButton!1; 


private ImageButton myButton2; 


private ImageButton myButton3; 


private ImageButton myButton4; 


private ListView myListView!l1; 


private String strTempFile = "ex07 11 "; 


private File myRecAudioFile; 


private File myRecAudioDir; 


private File myPlayFile; 
private MediaRecorder mMediaRecorder01; 


private ArrayList<String> recordFiles; 


private ArrayA dapter<String> adapter; 


private TextView myTextView!; 


private boolean sdCardExit; 


private boolean isStopRecord,; 


通过 findViewByld 分 别 构 造 4 个 按钮 对 象 和 2 个 文本 对 象 ,然后 设置 按钮 状态 不 可 选 。 
具体 代码 如 下 。 


/** Called when the activity is first created. */ 


(QOverride 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 

旋 设置 4 个 按钮 和 2 个 文本 沁 

myButtonl = (ImageButton) fndViewById(R.id.ImageButton01); 
myButton2 = (ImageButton) fndViewById(R.id.ImageButton02); 
myButton3 = (ImageButton) fndViewById(R.id.ImageButton03 ); 
myButton4 = (ImageButton) fndViewById(R.id.ImageButton04); 


myListViewl = 


(ListView) findViewById(R.id.ListView01); 


myTextViewl = (TextView) findViewById(R.id.TextView01); 
上 # 设置 按钮 状态 不 可 选 */ 

myButton2.setEnabled(false); 

myButton3.setEnabled(false); 

myButton4.setEnabled(false); 


取得 SD Card 


iT 


iT 


目录 里 的 所 有 .amr 格式 的 文人 


， 最 后 将 ArrayAdapter 添加 到 ListView 
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对 象 中 。 有 具体 代码 如 下 。 


入 判断 SD Card 是 否 插入 */ 
sdCardExit = Environment.getExternalStorageState().equals( 
android.o0s.Environment.MEDIA MOUNTED); 
雍 取得 SD Card 路 径 作为 录音 的 文件 位 置 沁 
if (sdCardExit) 
myRecAudioDir = Environment.getExternalStorageDirectory(); 


上 取得 SD Card 目录 里 的 所 有 .amr 格式 的 文件 */ 
getRecordFiles(); 


adapter = new ArrayA dapter<String>(this, 
R.layout.my_simple list item, recordFiles); 

族 将 ArrayAdapter 添加 ListView 对 象 中 */ 

ImyListViewl.setAdapter(adapter); 


4) 设置 单 击 “ 录 音 ” 按 钮 后 的 录音 处 理事 件 ， 流 程 如 下 。 
口 第 一 步 : 创建 录音 文件 。 
口 第 二 步 : 设置 录音 来 源 为 麦克 风 。 

口 第 三 步 : 通过 myTextViewl.setText(" 录 音 中 由 方法 设置 录音 过 程 显 示 的 提示 文本 。 
号 体 代码 如 下 。 


/# 年/ 
myButtonl.setOnClickListener(new ImageButton.OnClickListener() 
{ 
(@Override 
public void onClick(View arg0) 
{ 
try 
{ 
if (!sdCardExit) 
{ 
Toast.makeText(example9.this, "请 插入 SD Card"， 
Toast.LENGTH LONG).show(); 
return; 
} 
庆 创建 录音 文件 * 
myRecAudioFile = File.createTempFile(strTempFile, ".amr", 
myRecAudioDir); 
mMediaRecorder01 = new MediaRecorder(); 
庆 设置 录音 来 源 为 麦克 风 */ 
mMediaRecorder01 
.SetAudioSource(MediaRecorder.AudioSource.MIC); 
ImMediaRecorder01 
.SetOutputFormat(MediaRecorder.OutputFormat.DEFAULT); 
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ImMediaRecorder01 
.SetAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT); 

mMediaRecorder01.setOutputFile(myRecAudioFile 
.getAbsolutePath()); 

mMediaRecorder01.prepare(); 

mMediaRecorder0 1.start(); 

myTextView1l.setText(" 录 音 中 "); 

myButton2.setEnabled(true); 

myButton3.setEnabled(false); 

myButton4.setEnabled(false); 

isStopRecord = false; 


catch (IOException e) 


{ 


} 
} 


// TODO Auto-generated catch block 
e.printStackTrace(); 


5) 设置 单 击 “ 停 止 ”按钮 后 的 处 理事 件 ， 流 程 如 下 。 


口 


| 


第 


A 一 一 


ZU 
上 二 


口 叮 消 : 


通过 mMediaRecorder01.stop(0) 方 法 停止 录音 。 
将 录音 频 文件 名 传递 给 Adapter。 


体 代 码 如 下 。 


~ 


le 
myButton2.setOnClickListener(new ImageButton.OnClickListener() 


{ 


(QOverride 


public void onClick(View arg0) 


{ 


if (myRecAudioFile != null) 


{ 


入 停止 录音 % 

mMediaRecorder01.stop(); 

mMediaRecorder0 1 .release(); 

mMediaRecorder01 = null; 

庆 将 录音 频 文件 名 传 给 Adapter */ 
adapter.add(myRecAudioFile.getName()); 
myTextViewl.setText(" 停 止 : " +myRecAudioFile.getName()); 
myButton2.setEnabled(false); 

isStopRecord = true; 
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6) 设置 单 击 “ 播 放 ” 按 钮 后 的 处 理事 件 ， 


Oh/ 
myButton3.setOnClickListener(new ImageButton.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
if (myPlayFile != null && myPlayFile.exists()) 
{ 
此 打开 播放 的 程序 汶 
openFile(myPlayFile); 
} 
} 
»); 


7) 设置 单 击 “删除 ”按钮 后 的 处 理事 件 ， 流 程 如 下 。 
口 第 一 步 : 将 Adapter 删除 文件 名 。 

口 第 二 步 : 删除 存在 的 文件 。 

所 体 代码 如 下 。 


上 # 删除 沁 
myButton4.setOnClickListener(new ImageButton.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
if (myPlayFile != null) 
{ 
庆 先 将 Adapter 删除 文件 名 *%/ 
adapter.remove(myPlayFile.getName()); 
谍 删除 文件 *%/ 
if (myPlayFile.exists()) 
myPlayFile.delete(); 
myTextViewl.setText(" 完 成 删除 风 ; 


单 击 后 将 打开 播放 的 程序 。 有 具体 代码 如 下 。 


} 
)); 
8) 设置 单 击 列表 中 文件 的 处 理事 件 ， 流 程 如 下 。 
口 第 一 步 : 如 果 有 文件 ， 单 击 后 删除 这 个 文件 ， 并 将 播放 按钮 设置 为 不 可 用 。 
口 第 二 步 : 输出 选择 提示 语句 。 
体 代码 如 下 。 


ImyListViewl.setOnItemClickListener 
(new AdapterView.OnItemClickListener() 


350 看 看 


第 7 章 _ 娱 乐 和 多 媒体 编程 
{ 
Override 
public void onItemClick(AdapterView<?> arg0, View argl, 
int arg2, long arg3) 
{ 
myButton3.setEnabled(true); 
myButton4.setEnabled(true); 
myPlayFile = new File(myRecAudioDir.getAbsolutePath() 
+ File.separator 
+((CheckedTextView) arg1).getText()); 
myTextView1.setText(" 你 选 的 是 :" 
+((CheckedTextView) arg1).getText()); 


人 
) 


9) 定义 方法 onStop0， 用 于 停止 录音 处 理 。 具 体 代 人 码 如 下 。 


(@Override 
protected void onStop() 
{ 
if (mMediaRecorder01 != null && lisStopRecord) 
{ 
谨 停止 录音 */ 
mMediaRecorder01.stop(); 
mMediaRecorder01 .release(); 
mMediaRecorder01 = null; 
} 
super.onStop(); 


} 


10) 定义 方法 getRecordFiles()， 用 于 获取 文件 的 长 度 ， 并 设置 只 获取 “.amr” 格 式 的 文 


具体 代码 如 下 。 


private void getRecordFiles() 
{ 
recordFiles = new ArrayList<String>(); 
if (sdCardExit) 
{ 
File files[] = myRecAudioDir.listFiles(); 
if (files != null) 
{ 
for (int i= 0; i < files.length; 1++) 
{ 
if (files[i].getName().indexOf(".") >= 0) 
{ 
/# 只 获取 .amr 文件 六 
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String fileS = files[i].getName().substring( 
files[i].getName().indexOf(".")); 
if (fileS.toLowerCase().equals(".amr")) 
recordFiles.add(files[i].getName()); 


11) 定义 方法 openFile(File f， 用 于 打开 指定 的 录音 文件 。 具 体 代码 如 下 。 


入 打开 录音 文件 的 程序 */ 

private void openFile(File f) 

{ 
Intent intent = new Intent(); 
intent.addFlags(Intent.FLAG ACTIVITY NEW_ TASRK); 
intent.setAction(android.content.Intent.ACTION VIEW); 
String type = get MIMEType(f); 
intent.setDataAndType(Uri.fromFile(f), type); 
startActivity(intent); 


} 


12) 定义 方法 getMIMEType(File f, 用 于 获得 文件 的 类 型 , 在 此 设置 了 audio 类 型 、image 
类 型 和 其 他 类 型 ， 共 三 大 类 。 具 体 代码 如 下 。 


private String get MIMEType(File f) 
{ 

String end = f.getName().substring( 
f.getName().lastIndexOf(".") + 1, f.getName().length()) 
.toLowerCase(); 

String type = ""; 

if (end.equals("mp3") || end.equals("aac") || end.equals("aac" 
|| end.equals("amr") || end.equals("mpeg") 
|| end.equals("mp4")) 

type=" 

} 

else if (end.equals("jpg") || end.equals("gif") 

| end.equals("png") || end.equals("jpeg")) 


audio"; 


{ 
type = "image"; 
} 
else 
{ 
type We 
} 
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type += /*"; 
return type; 
} 
} 


最 后 ， 需 要 在 文件 AndroidManifest.xml 中 打开 录音 权限 。 具 体 代 码 如 下 。 


<uses-permission android:name="android.permission.RECORD AUDIO"> 


执行 后 的 效果 如 图 7-17 所 示 。 当 单 击 “ 录 音 ” 按 钮 时 开始 录音 处 理 ， 如 图 7-18 所 示 。 
当 单 击 “ 停 止 ” 按 钮 后 停止 录音 处 理 ， 并 在 列表 中 显示 录制 的 音频 文件 ， 如 图 7-19 所 示 ; 当 
选中 音频 文件 ， 单 击 “ 删 除 ” 按 钮 后 会 删除 选中 音频 文件 ， 单 击 “ 播 放 ” 按 钮 后 会 播放 选中 


的 音频 文件 ， 如 图 7-21 所 示 。 
4f 录音 中 1@ 停 |E IPE | 园 出 除 | 
录音 中 


图 7-17 初始 效果 图 7-18 正在 录音 
吗 国 锚 天 全 7:31w 


practice9 


区 录音 | | 人 @ 停 止 | 国志 5K| | 园 二 | 4f 录音 | © 停止 图 播放 || | 同 草 除 
停止 : ex07_11_52808.amr Ey 


ex07_11_52808.amr 完成 删除 


1 停止 | | 固 基 族 | 本 m 除 | 


图 7-19 显示 录制 的 文件 图 7-20 ”删除 音频 


园 动 天 揭 7:31m 


"ram 
pe pe ee en po ere ee ee pe 
a ls lo le le lah he 中 

re 


图 7-21 播放 音频 
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ey 


在 移动 手机 中 ， 拍 照 和 录制 视频 也 是 十 分 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 有 具 
体 实 例 的 实现 ， 介 绍 在 Android 手机 中 实现 相机 预览 和 拍照 处 理 的 基本 流程 。 本 实例 的 源 代 
人 码 保存 在 “光盘 :daimax7vexample10”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


7.10.1 实现 原理 
在 相机 中 ， 由 一 个 Preview 功能 ， 能 够 实现 预览 处 理 。 在 Preview 的 APIDemo 中 ， 没 有 
示范 如 何在 “Activity” 中 实现 Preview 的 讲解 ， 也 没有 提 到 如 何 获取 Preview 中 画面 的 方法 。 
本 实例 是 一 个 简单 的 拍照 练习 ， 和 API Demo 的 Preview 程序 不 同 。 实 例 中 将 以 Activity 
为 基础 ,在 Layout 中 配置 了 3 个 按钮 ， 分 别 实 现 打 开 预 览 、 关 闭 相机 和 拍照 处 理 这 3 个 功能 。 
当 单 击 “ 拍 照 ” 按 钮 后 ， 需 要 先 将 画面 截取 下 来 ， 并 存储 到 SD 卡 中 ， 并 将 拍 下 来 的 图 片 显 
示 在 Activity 中 的 ImageView 中 。 为 避免 拍照 相片 造成 的 存储 卡 垃圾 暂 存 堆栈 ， 在 离开 程序 
前 将 临时 文件 删除 。 


7.10.2 ”编程 思想 
目前 的 智能 手机 拥有 很 多 强大 的 功能 ， 例 如 摄像 尖 、GPS 和 无 线 上 网 等 ， 现 在 是 开始 充 
使 用 这 些 功 能 的 时 候 了 ， 在 本 篇 文章 中 我 们 一 起 学 习 ， 如 何在 Google Android 编程 环境 中 ， 
单 的 方式 实现 Google Android 摄像 头 拍照 。 
注意 : 从 Android 1.5 (代号 cupcake ) 版 本 后 ， 在 安全 方面 有 诸多 改进 ， 其 中 之 一 与 摄像 
头 权限 控制 有 关 。 在 此 之 前 ， 能 够 创建 无 需 用 户 许可 就 可 实现 拍照 的 应 用 。 现 在 该 问题 已 被 
修复 ， 如 果 想 在 自己 的 应 用 中 使 用 摄像 头 ,需要 在 文件 AndroidManifest.xml 中 增加 以 下 代码 。 


世 


江宁 


荆 
加 


三 | 
到 


<uses-permission android:name="android.permission.CAMERA"/> 


1. 设 定 摄像 头 布局 
这 是 开发 工作 的 基础 ， 也 就 是 我 们 希望 在 应 用 程序 中 增加 多 少 辅助 性 元 素 ， 如 摄像 头 、 
各 种 功能 按钮 等 。 本 例 采 取 最 简 方式 ， 除 了 拍照 外 ， 没 有 多 余 功能 。 下 面 一 起 看 一 下 示例 将 


要 用 到 的 布局 文件 “camera_surface.xml”。 


<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
androlid:layout width="fill parent" android:layout_ height="fill parent" 
android:orientation="vertical"> 

<SurfaceView android:id="(@+id/surface camera" 
android:layout_width="fill parent" android:layout_ height="10dip" 
android:layout_ weight="1"> 

</SurfaceView> 

</LinearLayout> 


在 上 述 过 程 ， 不 能 在 资源 文件 名 称 中 使 用 大 写字 母 ， 如 果 把 该 文件 命名 为 “CameraSur- 
face.xml”， 会 之 来 不 必要 的 麻烦 。 
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该 布局 非常 简单 ， 只 有 一 个 LinearLayout 视 


面 ) 视图 ， 也 就 是 摄像 头 屏幕 。 


2. 摄像 头 实现 代码 


下 面 再 来 看 一 下 Android 代码 。 创 建 一 


SurfaceHolderCallback 接口 。 


后 


组 ， 
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I 


中 只 有 一 个 SurfaceView〔 可 画图 界 


个 名 为 “CameraView” 的 Activity 类 ， 实 现 


public class CamaraView extends Activity implements SurfaceHolder.Callback 


接口 SurfaceHolder.Callback 被 ) 
口 surfaceChanged: 当 预 览 界面 的 格式 和 大 小 发 生 改 变 时 ， 
口 surfaceCreated: 初次 实例 化 ， 预 览 界 面 被 创建 时 ， 该 方法 被 调用 。 
口 surfaceDestroyed: 当 预 览 田 


下 面 看 一 下 在 摄像 头 应 


super.onCreate(icicle); 


] 中 


面 被 关闭 时 ， 
如 何 使 用 这 个 接口 


该 方法 被 调用 。 
， 首 先 看 一 下 在 Activity 类 中 的 onCreate 


来 接收 摄像 头 预 览 界面 变化 的 信息 。 它 实现 了 三 个 方法 。 


getWindow().setFormat(PixelFormat.TRANSLUCENT); 

requestWindowFeature(Window.FEATURE NO TITLE); 
getWindow!().setFlags(WindowManager.LayoutParams.FLAG FULLSCREEN, 
WindowManager.LayoutParams.FLAG FULLSCREEN); 
setContentView(R.layout.camera); 
mSurfaceView = (SurfaceView) findViewByld(R.id.surface camera); 
mSurfaceHolder = mSurfaceView.getHolder(); 
mSurfaceHolder.addCallback(this); 
mSurfaceHolder.setType(SurfaceHolder.SURFACE TYPE PUSH BUFFERS); 


} 


下 面 逐 一 对 代码 进行 说 明 。 


getWindow().setFormat(PixelFormat.TRANSLUCENT); 

requestWindowFeature(Window.FEATURE NO TITLE); 
getWindow().setFlags(WindowManager.LayoutParams.FLAG FULLSCREEN, 
WindowManager.LayoutParams.FLAG FULLSCREEN); 


Ft 


通过 上 述 代码 ， 读 者 告诉 


六 


Ff 幕 两 点 信 居 。 


1) 摄像 头 预览 界面 将 通过 全 屏 显 示 ， 没 有 “标题 (title)”。 


2) 屏幕 格式 为 “ 半 透 明 ”。 


setContentView(R.layout.camera surface ); 


mSurfaceView = (SurfaceView) findViewByld(R.id.surface camera); 


并 创建 一 个 SurfaceView 对 象 ， 从 XML 文件 


中 获得 布 


mSurfaceHolder = mSurfaceView.getHolder(); 
mSurfaceHolder.addCallback(this); 


该 方法 被 调用 。 


T 


在 以 上 代码 中 ,读者 通过 setContentView 设 定 Activity 的 布局 为 前 


局 信 息 。 


面 创建 的 camera_surface， 
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mSurfaceHolder.setType(SurfaceHolder.SURFACE TYPE PUSH _ BUFFERS); 


通过 以 上 代码 , 读者 从 surfaceview 中 获得 了 holder, 并 增加 callback (回放 ) 功 能 到 “this”。 


这 意味 着 我 们 的 操作 (Activity) 将 可 以 管理 这 个 surfaceview。 
看 一 下 callback 功能 时 如 何 实现 的 。 


public void surfaceCreated(SurfaceHolder holder) { 
mCamera = Camera.open(); 


mCamera 是 “Camera” 类 的 一 个 对 象 。 在 surfaceCreated 方法 中 “ 打 
是 启动 它 的 方式 。 


public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) { 
if (mPreviewRunning) { 

mCamera.stopPreview(); 

} 

Camera.Parameters p = mCamera.getParameters(); 
p.setPreviewSize(w, h); 
mCamera.setParameters(p); 

ty { 

mCamera.setPreviewDisplay(holder); 

} catch (IOException e) { 

e.printStackTrace(); 

} 

mCamera.startPreview(); 

mPreviewRunning = true; 


} 


”摄像 头 ， 这 就 


该 方法 让 摄像 头 做 好 拍照 准备 ， 设 定 它 的 参数 ， 并 开始 在 Android 屏幕 
本 例 使 用 了 一 个 “semaphore” 参 数 来 防止 冲突 : 当 mPreviewRunning 为 true 
头 处 于 激活 状态 ， 并 未 被 关闭 ， 因 此 可 以 使 用 它 。 


public void surfaceDestroyed(SurfaceHolder holder) { 
mCamera.stopPreview(); 

mPreviewRunning = false; 

mCamera.release(); 


} 


通过 这 个 方法 ， 停 止 摄像 头 并 释放 相关 的 资源 。 正 如 大 家 所 看 到 的 ， 在 此 设置 


局 动 预览 画面 。 
时 ， 意 味 着 摄像 


mPreviewRunning 为 false， 以 此 来 防止 在 surfaceChanged 方法 中 的 冲突 。 原 


因 何 在 ? 因为 这 


意味 着 用 户 已 经 关闭 了 摄像 类 ， 而 且 不 能 再 设置 其 参数 或 在 摄像 关中 局 动 图 像 预 览 。 


最 后 看 一 下 本 例 中 最 重要 的 方法 。 


Camera.PictureCallback mPictureCallback = new Camera.PictureCallback() { 
public void onPictureTaken(byte[| imageData, Camera c) { 
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当 拍 照 时 ， 该 方法 被 调用 。 举 例 来 说 ， 读 者 可 以 在 界面 上 创建 一 个 OnClickListener 方法 ， 
单 击 屏幕 时 ， 调 用 PictureCallBack 方法 。 这 个 方法 会 提供 图 像 的 字 节 数组 ， 然 后 你 可 以 使 用 
Android 提供 的 Bitmap 和 BitmapFactory 类 ， 将 其 从 字 节 数组 转换 成 你 想 要 的 图 像 格式 。 


7.10.3 ”具体 实现 

本 实例 的 主 程序 文件 是 example10.java， 下 面 开始 讲解 其 具体 实现 代码 。 

1) 先 使 Activity 实现 SurfaceHolder.Callback， 创 建 私 有 Camera 对 象 ， 然 后 分 别 创建 
mlmageView01、mTextView01、TAG、mSurfaceView01、mSurfaceHolder01 和 intScreenY， 作 
为 review 照 下 来 的 照片 之 用 。 有 具体 代码 如 下 。 


iT 


/# 使 Activity 实现 SurfaceHolder.Callback */ 
public class example10 extends Activity 
implements SurfaceHolder.Callback 
{ 

/# 创建 私有 Camera 对 象 */ 


private Camera mCamera01; 


private Button mButton01, mButton02, mButton03; 


/#z 作为 review 照 下 来 的 相片 之 用 */ 
private ImageView mImageView0!1; 


private TextView mTextView01; 

private String TAG = "HIPPO"; 

private SurfaceView mSurfaceView0!1; 
private SurfaceHolder mSurfaceHolder01; 
//private int intScreenX, intScreenY; 


2) 默认 相机 预览 模式 为 false， 将 照 下 来 的 图 片 存储 在 “/sdcard/camera_snap.jpg”。 具体 
代码 如 下 。 


默认 相机 预览 模式 为 false */ 


private boolean blfPreview = false; 


庆 将 照 下 来 的 图 片 存储 在 此 */ 
private String strCaptureFilePath = "/sdcard/camera snap.jpg"; 


3) 通过 requestWindowFeature(Window.FEATURE NO _TITLE)， 设 置 程序 全 屏幕 运行 ， 
而 不 使 用 标题 栏 。 然 后 判断 存储 卡 是 否 存在 ， 并 提醒 用 户 未 安装 存储 卡 。 有 具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


入 应 用 程序 全 屏幕 运行 ， 不 使 用 标题 栏 */ 
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requestWindowFeature(Window.FEATURE NO_TITLE); 
setContentView(R.layout.main); 


/* 判断 存储 卡 是 否 存在 * 
if(!check SDCard()) 
{ 
/* 提醒 User 未 安装 存储 卡 */ 
mMakeTextToast 
( 


getResources().getText(R.string.str_ err nosd).toString(), 


true 
); 
) 


4) 通过 DisplayMetrics 对 象 dm 取得 屏幕 解析 像素 , 然后 以 SurfaceView 作为 相机 预览 之 


并 绑 定 SurfaceView， 取 得 SurfaceHolder 对 象 ， 最 后 通过 setFixedSize 额外 设置 预览 大 小 。 
具体 代码 如 下 。 


上 取得 屏幕 解析 像素 */ 
DisplayMetrics dm = new DisplayMetrics(); 


getWindowManager().getDefaultDisplay().getMetrics(dm); 
intScreenX = dm.widthPixels; 

intScreenY = dm.heightPixels; 

Log.i(TAG, Integer.toString(int ScreenX)); 


mTlextView01 = (TextView) fndViewById(R.id.myTextViewl); 
mlmageView01 = (ImageView) findViewById(R.idmyImageViewl); 


/# 以 SurfaceView 作为 相机 Preview 之 用 */ 
mSurfaceView01 = (SurfaceView) findViewById(R.id.mSurfaceViewl); 


/# 绑 定 SurfaceView， 取 得 SurfaceHolder 对 象 */ 
mSurfaceHolder01 = mSurfaceView01.getHolder(); 


Activity 必须 实现 SurfaceHolder.Callback */ 
mSurfaceHolder01.addCallback(example10.this); 


谍 预览 大 小 设置 ， 在 此 不 使 用 */ 
mSurfaceHolder01.setFixedSize(320, 240); 


/* 

* 以 SURFACE TYPE PUSH BUFFERS(3) 

* 作为 SurfaceHolder 显示 类 型 

米 */ 
mSurfaceHolder01.setType 
(SurfaceHolder.SURFACE TYPE PUSH BUFFERS); 
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mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton02 = (Button)findViewById(R.id.myButton2); 
mButton03 = (Button)findViewById(R.id.myButton3); 


5) 设置 打开 相机 及 浏览 按钮 事件 ， 自 定义 初始 化 打开 相机 方法 。 具 体 代码 丸 


入 打开 相机 及 浏览 六 
mButton01.setOnClickListener(new Button.OnClickListener() 


{ 


(QOverride 
public void onClick(View arg0) 


{ 


WW 
ini 
} 
)); 


6) 设置 停 | 


如 下 。 


自 定义 初始 化 打 


tCamera(); 


入 停止 及 浏览 相机 */ 
mButton02.setOnClickListener(new Button.OnClickListener() 


{ 


(QOverride 
public void onClick(View arg0) 


{ 


We 


resetCamera(); 


} 
小 


7) 设置 停 ] 


照 方 法 takePicture()。 


上 拍照 按钮 事件 ， 
L 体 代码 如 下 。 


入 拍照 7 
mButton03.setOnClickListener(new Button.OnClickListener() 


{ 


Override 
public void onClick(View arg0) 


{ 


相机 方法 */ 


上 


自 定义 重 置 相机 ， 并 关闭 相机 预览 方法 光 


1 


上 浏览 及 相机 按钮 事件 ， 自 定义 重 置 相机 ， 并 关闭 相机 预览 方法 。 具 体 代 码 


当 存 储 卡 存在 才 允 许 拍 照 ， 存 储 暂 存 图 像 文件 ， 


启 如 果 存 储 卡 存在 才 人 允许 拍照 ， 存 储 暂 存 图 像 文件 */ 
if(checkSDCard()) 


{ 


诈 自 定义 拍 
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上 # 自 定义 拍照 方法 */ 
takePicture(); 

} 


else 

{ 
旋 存储 卡 不 存在 显示 提示 */ 
ImTextView01.setText 


( 


getResources().getText(R.string.str err nosd).toString() 


8) 定义 initCamera0 方 法 ， 若 相机 在 非 预览 模式 ， 则 打开 相机 。 流 程 如 下 。 
口 第 一 步 : 创建 Camera.Parameters 对 象 。 
口 第 二 步 : 设置 照片 格式 为 JPEG。 

口 第 三 步 ， 指定 preview 的 屏幕 大 小 。 
口 

口 

口 


第 四 步 : 设置 图 片 分 辩 率 大 小 。 
第 五 步 : 将 Camera.Parameters 参数 设置 为 Camera。 
TA 


第 六 步 : 运行 预览 模式 。 
人体 代码 如 下 。 


旋 自 定义 初始 相机 方法 六 
private void initCamera() 
{ 
if(!bIfPreview) 
{ 
入 若 相 机 在 非 预 览 模式 ， 则 打开 相机 */ 


mCamera01 = Camera.open(); 


} 


if (mCamera01 !=null && !blfPreview) 


{ 
Log.i(TAG, "inside the camera"); 


/* 创建 Camera.Parameters 对 象 */ 
Camera.Parameters parameters = mCamera01.getParameters(); 


/* 设置 相片 格式 为 JPEG */ 


parameters. setPictureFormat(PixelFormat.JPEG); 


/* 指定 preview 的 屏幕 大 小 */ 
parameters.setPreviewSize(320, 240); 
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旋 设置 图 片 分 辩 率 大 小 */ 
parameters. setPictureSize(320, 240); 


/* 将 Camera.Parameters 设置 给 Camera */ 


mCamera01.setParameters(parameters); 


/* setPreviewDisplay 唯一 的 参数 为 SurfaceHolder */ 
mCamera01.setPreviewDisplay(mSurfaceHolder01); 


/* 立即 运行 Preview */ 
mCamera01.startPreview(); 
bIfPreview = true; 


} 


9) 定义 takePicture0 方 法 ， 调 用 takePicture0 方 法 拍照 并 截取 图 像 。 具 体 代 人 码 如 下 。 
旋 拍照 截取 图 像 尝 


private void takePicture() 
{ 
if (mCamera01 != null && bIfPreview) 
{ 
/* 调用 takePicture0 方 法 拍照 */ 
mCamera01.takePicture 
(shutterCallback, rawCallback, jpegCallback); 
} 
} 


10) 定义 resetCamera() 方 法 ， 实 现 相 机 重 置 。 具 体 代码 如 下 。 
从 相机 重 置 * 


private void resetCamera() 

{ 
if (mCamera01 != null && bIfPreview) 
{ 


mCamera01.stopPreview(); 
/# 扩展 学 习 ， 释 放 Camera 对 象 */ 
mCamera01 .release(); 


wh 


mCamera01 = null; 


bIfPreview = false; 


private ShutterCallback shutterCallback = new ShutterCallback() 
{ 


public void onShutter() 
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{ 
} 
}; 
private PictureCallback rawCallback = new PictureCallback() 
{ 
public void onPictureTaken(byte[] _data, Camera camera) 
{ 
} 
}; 
11) 通过 onPictureTaken 对 象 传 入 的 第 一 个 参数 表示 相片 的 大 小 ， 并 用 new File(strCapture 


i 


FilePath) 创建 一 个 新 文件 ， 并 采用 压缩 转 档 方法 压缩 图 片 ， 然 后 调用 flush0 方 法 更 新 
BufferStream Re ， 最 后 将 拍照 下 来 且 存 储 完毕 的 图 像 显示 出 来 。 具 体 代 码 如 下 。 


private PictureCallback jpegCallback = new PictureCallback() 
{ 


public void onPictureTaken(byte[] _data, Camera camera) 


{ 


/* onPictureTaken 传 入 的 第 一 个 参数 即 为 相片 的 byte */ 
Bitmap bm = BitmapFactory.decodeByteArray 
(data, 0，data.length); 


入 创建 新 文件 沁 

File myCaptureFile = new File(strCaptureFilePath); 
try 

{ 


BufferedOutputStream bos = new BufferedOutputStream 


(new FileOutputStream(myCaptureFile)); 


T 


入 采用 压缩 转 档 方法 */ 
bm.compress(Bitmap.CompressFormat.JPEG, 80, bos); 


/* 调用 flush() 方 法 ， 更 新 BufferStream */ 
bos.flush(); 


/# 结束 OutputStream */ 
bos.close(); 


IC 


放 将 拍照 下 来 且 存 储 完毕 的 图 文件 ， 显 示 出 来 光 
mlmageView01.setImageBitmap(bm); 


ta 


相机 ， 并 关闭 预览 岂 


作 显示 完 图 文件 ， 立 即 习 
resetCamera(); 
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从 再 重新 启动 相机 继续 预览 凡 
initCamera(); 

} 

catch (Exception e) 

{ 
Log.e(TAG, e.getMessage()); 

} 

} 
上 


上 


12) 通过 自 定义 方法 delFile(String strFileName)， 删 除 临时 文件 ， 具 体 代码 如 下 。 


旋 自 定 义 删除 文件 方法 * 
private void delFile(String strFileName) 
{ 

try 

{ 


File myFile = new File(strFileName); 
if(myFile.exists()) 
{ 
myFile.delete(); 

} 

} 

catch (Exception e) 

{ 
Log.e(TAG, e.toString()); 
e.printStack Trace(); 

} 

} 


13) 通过 方法 mMakeTextToast(String str, boolean isLong), 输出 提示 语句 。 具 体 代码 如 下 。 


public void mMakeTextToast(String str, boolean isLong) 
{ 
if(isLong==true) 
{ 
‘Toast.makeText(example10.this, str, Toast.LENGTH LONG).show(); 
} 
else 
{ 
Toast.makeText(example10.this, str, ToastLENGTH SHORT).show(); 
} 
} 


14) 通过 方法 checkSDCard()， 检 查 是 否 有 存储 卡 。 具 体 代 码 如 下 。 


private boolean checkSDCard() 


{ 
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谍 判断 存储 卡 是 否 存 在 */ 
if(android.os.Environment.getExternalStorageState().equals 
(android.o0s.Environment.MEDIA MOUNTED)) 
{ 


return true; 


} 


else 


{ 


return false; 


(QOverride 
public void surfaceChanged 
(SurfaceHolder surfaceholder, int format, int w, int h) 
{ 
Log.i(TAG, "Surface Changed"); 
} 


(QOverride 
public void surfaceCreated(SurfaceHolder surfaceholder) 


{ 
Log.i(TAG, "Surface Changed"); 


} 


片 ， 其 体 代码 如 下 。 


ES 


15〉 当 Surface 不 存在 时 需要 删除 


(QOverride 
public void surfaceDestroyed(SurfaceHolder surfaceholder) 
{ 
/* 如 果 Surface 不 存在 
try 
{ 
delFile(strCaptureFilePath); 
catch(Exception e) 
{ 


e.printStack Trace(); 


} 
Log.i(TAG, "Surface Destroyed"); 


需要 删除 图 片 * 


} 


接 下 来 ， 需 要 在 文件 AndroidManifest.xml 中 设置 android.permission.CAMERA 权限 ， 具 
体 代 码 如 下 。 


<uses-permission android:name="android.permission.CAMERA"> 
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执行 后 的 效果 如 图 7-22 所 示 。 单 击 “ 预 览 ”、“ 拍 照 ”和 “关闭 ”按钮 后 ， 能 够 实现 对 应 
的 功能 
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图 7-22 执行 效果 
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在 移动 手机 中 ， 录 音 处 理 也 是 十 分 重要 的 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 
的 实现 ， 介 绍 在 Android 手机 中 实现 录音 处 理 的 基本 流程 。 本 实例 的 源 代 码 保存 在 “ 光 
得 :daimav7\example11”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


7.11.1 实现 原理 

在 Android 中 ， 内 置 了 VideoView〔 视 频 视 图 组 件 ) 作 为 多 媒体 视频 播放 器 ， 这 样 可 以 浏 

览 电 影视 频 。VideoView 和 前 面 介绍 的 Widget 使 用 方法 类 似 ， 必 须 先 在 Layout XML 中 定义 

VideoView 属性 ， 在 程序 中 通过 findViewById0 方 法 即 可 创建 VideoView 对 象 。 
在 本 实例 中 ， 预 先 准 备 了 2 个 .3gp 格式 的 视频 文件 ， 上 传 到 了 虚拟 SD 卡 中 。 然 后 插入 2 

个 按钮 ， 当 单 击 按钮 后 分 别 实现 对 这 2 个 视频 文件 的 播放 。 


7.11.2 ”具体 实现 
本 实例 的 主 程序 文件 是 examplell.java， 下 面 开始 讲解 其 具体 实现 代码 。 
1) 先 加 载 相关 类 ， 然 后 分 别 声明 变量 mTextView01、mVideoView01、 strVideoPath、 


mButton01、mButton02 和 TAG。 有 具体 代码 如 下 。 


package irdc.examplell; 


import irdc.examplel1.R; 
import android.app.Activity; 
import android.graphics.PixelF ormat; 
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import android.media.MediaPlayer; 
import android.net.Uri; 
import android.os.Bundle; 
import android.util.Log; 
import android.view.View; 
import android.widget.Button; 
import android.widget.MediaController; 
import android.widget.TextView; 
import android.widget.Toast; 
import android.widget.VideoView; 


public class examplel 1 extends Activity 
{ 
private TextView mTextView01; 
private VideoView mVideoView01; 
private String strVideoPath = ""; 


private Button mButton01, mButton02; 
private String TAG = "HIPPO VIDEOVIEW"; 


2) 先 设置 判别 是 否 安装 存储 卡 变量 flag (标识 ) 为 false， 然 后 设置 全 屏幕 显示 。 有 具体 代 


/# 默认 判别 是 否 安装 存储 卡 flag 为 false */ 
private boolean bIfSDExist = false; 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


谨 全 屏幕 沁 
getWindow().setFormat(PixelFormat.TRANSLUCENT); 
setContentView(R.layout.main); 


3) 判断 存储 - 
如 下 。 


是 否 存在 ， 不 存在 则 通过 mMakeTextToast 对 象 输出 提示 语句 。 有 具体 代码 


AT 


1* 判断 存储 卡 是 否 存 在 忆 
if(android.os.Environment.getExternalStorageState().equals 
(android.os.Environment.MEDIA MOUNTED)) 
{ 
bIfSDExist = true; 
} 
else 
{ 
bIfSDExist = false; 


366 看 


第 7 章 _ 娱 乐 和 多 媒体 编程 


mMakeTextToast 
( 


getResources().getText(R.string.str err nosd).toString(), 
true 


); 
} 


4) 分 别 取 得 TextView、EditText， 然 后 通过 findViewByld 构造 2 个 对 象 。 具体 代 码 如 下 。 


mTlextView01 = (TextView)findViewBylId(R.id.myTextView!]); 
mVideoView01 = (VideoView)findViewById(R.id.myVideoView1l); 


mButton01 = (Button)findViewById(R.id.myButton!); 
mButton02 = (Button)findViewById(R.id.myButton2); 


5) 定义 单 击 处 理事 件 ， 通 过 playVideo(strVideoPath) 方 法 播放 第 一 个 影片 A。 


体 代 码 如 下 。 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 
public void onClick(View arg0) 
{ 
if(bIfSDExist) 
{ 
席 播放 影片 路 径 1 */ 
strVideoPath = "file:///sdcard/hello.3gp"; 
playVideo(strVideoPath); 
) 
} 
); 


6) 定义 单 击 处 理事 件 ， 通 过 playVideo(strVideoPath) 方 法 播放 第 二 个 影片 B。 具 体 代码 
如 下 。 


mButton02.setOnClickListener(new Button.OnClickListener() 
{ 
Override 
public void onClick(View arg0) 
{ 
if(bIfSDExist) 
{ 
席 播放 影片 路 径 2 */ 
strVideoPath = "file:///sdcard/test.3gp"; 
playVideo(strVideoPath); 
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7) 自 定 义 VideoView0 方 法 ， 用 于 播放 指定 路 径 的 影片 。 有 具体 代 码 如 下 。 


/* 自 定义 以 VideoView 播放 影片 */ 
private void playVideo(String strPath) 
{ 
if(strPath!="") 
{ 
/# 调用 VideoURI 方 法， 指定 解析 路 径 * 
mVideoView01.setVideoURI(Uri.parse(strPath)); 


/* 设置 控制 Bar 显示 于 此 Context 中 */ 
mVideoView01.setMediaController 
(new MediaController(examplel 1.this)); 


mVideoView01.requestFocus(); 


/* 调用 VideoView.startO 自动 播放 */ 

mVideoView01.start(); 

if{mVideoView01.isPlaying()) 

{ 
/* 程序 不 会 被 运行 ， 因 start0 后 尚 需要 preparing() */ 
mTextView01.setText("Now Playing:"+strPath); 
Log.i(TAG, strPath); 

} 


} 
} 


8) 定义 mMakeTextToast(String str, boolean isLong) 方 法 ， 输 出 提醒 语句 。 


如 下 。 


public void mMakeTextToast(String str, boolean isLong) 
{ 
if(isLong==true) 
{ 
Toast.makeText(examplel 1.this, str, Toast.LENGTH LONG).show(); 
} 


else 


{ 
Toast.makeText(examplel1.this, str, Toast.LENGTH SHORT).show(); 


} 
} 
} 


执行 后 的 效果 如 图 7-23 所 示 。 当 单 击 
按钮 后 ， 分 别 播放 预 设 的 影片 。 
368 国 国 
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“播放 SD 3gp 影片 1” 和 “播放 SD 3gp 影片 2” 
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图 7-23 ”执行 效果 


/20000 


在 移动 手机 中 ,铃声 设置 也 是 十 分 重要 的 功能 ， 用户 可 以 去 网 络 中 下 载 自己 喜欢 的 铃声 。 
在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 手机 中 设置 指定 铃声 的 基本 
流程 。 本 实例 的 源 代码 保存 在 “光盘:\daima\N\example12”， 下面 开始 讲解 本 实例 的 具体 实现 


7.12.1 实现 原理 


在 Android 中 ， 通 过 ER 类 来 专门 控制 各 种 铃声 。 例 如 ， 常 见 的 来 电 铃声 、 
亲 钟 铃声 、 一 些 警告 和 信息 通知 。RingtoneManager 类 的 常用 方法 如 下 。 

口 getActualDefaultRingtoneUri: 获取 指定 类 型 的 当前 默认 铃声 。 

口 getCursor: 返回 所 有 可 用 铃声 的 游标 。 

口 getDefaultType: 获取 指定 URL 默认 的 铃声 类 型 。 
口 getDefaultUri: 返回 指定 类 型 默认 铃声 的 URL。 
口 getRingtoneUri: 返回 指定 位 置 铃声 的 
口 
口 
口 
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getRingtonePosition: 获取 指定 铃声 的 位 置 。 
getValidRingtoneUri: 获取 一 个 可 用 铃声 的 位 置 。 
isDefault: 获取 指定 URL 是 否 是 默认 的 铃声 。 

口 setActualDefaultRingtoneUri: 设置 默认 的 铃声 。 

在 Android 系统 中 ， 默 认 的 铃声 存储 在 “system/medio/audio” 目 录 中 ， 而 下 载 的 铃声 一 
般 被 保存 在 SD 卡 中 ， 假 设 下 载 的 铃声 分 别 保存 在 SD 卡 的 下 述 目录 中 。 

口 sdcard/music/ringtone: 一 般 来 电 铃 声 。 

口 sdcard/music/alarm: 曾 钟 铃声 

口 sdcard/music/notification: 警告 、 通 知 铃声 。 
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7.12.2 具体 实现 


编写 成 程序 Activityjava， 有 具体 实现 流程 如 下 : 
1) 分 别 定 义 3 个 按钮 、3 个 自 定 义 的 类 型 ， 设 置 3 个 铃声 文件 夹 。 具 体 代 码 如 下 。 


public class Activity01 extends Activity 

{ 
(HH 
private Button mButtonRingtone; 
private Button mButtonAlarm; 
private Button mButtonNotification; 


帮 目 是 光 的 类 型 品 


public static final int ButtonRingtone =0; 
public static final int ButtonAlarm =1; 
public static final int ButtonNotification =2; 


启 铃声 文件 夹 % 


private String strRingtoneFolder = "/sdcard/music/ringtone"; 


private String strAlarmFolder = "/sdcard/music/alarm"; 
private String strNotificationFolder = "/sdcard/music/notification"; 


2) 设置 单 击 按钮 mButtonRingtone 后 的 处 理事 件 ， 打 开 系 统 铃声 设置 ， 然 后 进行 设置 。 
具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButtonRingtone = (Button) fndViewById(R.id.ButtonRingtone); 
mButtonAlarm = (Button) findViewById(R.id.ButtonAlarm); 
mButtonNotification = (Button) findViewById(R.id.ButtonNotification); 
入 设置 来 电 铃声 * 
mButtonRingtone.setOnClickListener(new Button.OnClickListener() 
{ 
@Override 
public void onClick(View arg0) 
{ 
if (bFolder(strRingtoneFolder)) 
{ 


* 打 开 系 统 铃声 设置 */ 
Intent intent = new Intent(RingtoneManager.ACTION RINGTONE PICKER); 
/# 类 型 为 来 电 RINGTONE 模式 */ 
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intent.putExtra(RingtoneManager. EXTRA RINGTONE TYPE, RingtoneManager. 
TYPE RINGTONE); 


族 设 置 显示 的 标题 */ 
intentputExtra(RingtoneManagerEXTRA_RINGTONE TITLE, "设置 来 电 铃 声 "); 
/#* 当 设置 完成 之 后 返回 到 当前 的 Activity*/ 

startActivityForResult(intent, ButtonRingtone); 


由， 


3) 设置 单 击 按钮 mButtonAlarm 后 的 处 理 
体 代 码 如 下 。 


访 滩 管 六 名 从 产 秒 
mButtonAlarm.setOnClickListener(new Button.OnClickListener() 


{ 


i 


事件 ， 打 开 系统 铃声 设置 ， 然 后 进行 设置 。 具 


(@Override 
public void onClick(View arg0) 


{ 
if (bFolder(strAlarmFolder)) 


{ 


* 打 开 系 统 铃声 设置 */ 

Intent intent = new Intent(RingtoneManager.ACTION RINGTONE PICKER); 

任 设置 铃声 类 型 和 title*/ 
intent.putExtra(RingtoneManager. EXTRA RINGTONE TYPE, RingtoneManager.TYPE 


ALARM); 


intent.putExtra(RingtoneManager.EXTRA RINGTONE TITLE, "设置 闵 铃 铃声 "); 
必 当 设置 完成 之 后 返回 到 当前 的 Activity*/ 
startActivityForResult(intent, ButtonAlarm); 


1 


4) 设置 单 击 按钮 mButtonNotification 后 的 处 理 
具体 代码 如 下 。 


从 设置 通知 铃声 学 
mButtonNotification.setOnClickListener(new Button.OnClickListener() 


{ 


汕 


有 件 ， 打 开 系 统 铃声 设置 , 然后 进行 设置 。 


(@Override 
public void onClick(View arg0) 


{ 
if (bFolder(strNotificationFolder)) 


{ 


人 * 打 开 系 统 铃声 设置 */ 
Intent intent = new Intent(RingtoneManager.ACTION RINGTONE PICKER); 
画 国 371 
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人 # 设 置 铃声 类 型 和 title*/ 
intent.putExtra(RingtoneManager. EXTRA RINGTONE TYPE, RingtoneManager.TYPE 


NOTIFICATION); 
intent.putExtra(RingtoneManager.EXTRA _ RINGTONE_TITLE, "设置 通知 铃声 "); 
旋 当 设置 完成 之 后 返回 到 当前 的 Activity*/ 
startActivityForResult(intent, ButtonNotification); 

} 
} 
D); 
} 


5) 定义 方法 onActivityResult()， 此 方法 作为 设置 铃声 之 后 的 回调 方法 。 具 体 代码 如 下 。 


从 当 设 置 铃声 之 后 的 回调 方法 六 
(@Override 
protected void onActivityResult(int requestCode, int resultCode, Intent data) 
{ 
if (resultCode != RESULT OK) 
{ 
return; 
} 
switch (requestCode) 
{ 
case ButtonRingtone: 
try 
{ 


人 得 到 选择 的 铃声 */ 
Uri pickedUri = data.getParcelableExtra(RingtoneManager.EXTRA RINGTONE 


PICKED URD; 
/将 选择 的 铃声 设置 成 为 默认 六 
if (pickedUri != null) 
{ 


RingtoneManager.setActualDefaultRingtoneUri(Activity01.this, Ringtone 
Manager.TYPE RINGTONE, pickedUri); 


} 
} 
catch (Exception e) 
{ 
} 
break; 
case ButtonAlarm: 
try 
{ 
们 得 到 选择 的 铃声 允 


Uri pickedUri = data.getParcelableExtra(RingtoneManager.EXTRA RINGTONE 
PICKED URD; 
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放 将 选择 的 铃声 设置 成 为 默认 */ 
if (pickedUri != null) 
{ 


RingtoneManager.setActualDefaultRingtoneUri(Activity01.this, Ringtone 
Manager.TYPE ALARM, pickedUri); 
} 
} 
catch (Exception e) 
{ 


} 
break; 


case ButtonNotification: 
try 
{ 
人 # 得 到 选择 的 铃声 六 
Uri pickedUri = data.getParcelableExtra(RingtoneManager.EXTRA RINGTONE 


PICKED UR]); 
/# 将 选择 的 铃声 设置 成 为 默认 并 
if(pickedUri != null) 

{ 


RingtoneManager.setActualDefaultRingtoneUri(Activity01.this, Ringtone 
Manager.TYPE NOTIFICATION, pickedUri); 
} 
} 


catch (Exception e) 


{ 


} 
break; 


} 


super.onActivityResult(requestCode, resultCode, data); 


} 
6) 定义 方法 boolean bFolder， 用 于 检测 是 否 存 在 指定 的 文件 夹 ， 如 果 不 存在 则 创 于 
体 代码 如 下 。 


private boolean bFolder(String strFolder) 


{ 


次 
o 

上 
MA 


boolean btmp = false; 
File f= new File(strFolder); 
if (!f.exists()) 
{ 
if (f.mkdirs()) 
{ 


btmp = true; 


else 
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btmp = false; 


btmp = true; 
} 


return btmp; 


} 
执行 后 的 效果 如 图 7-24 所 示 。 在 此 可 以 分 别 设置 这 三 种 类 型 的 铃声 。 
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择 下 面 按钮 来 设置 铃声 


设置 来 电 铃声 


设置 闹钟 铃声 
设置 通知 铃声 


图 7-24 执行 效果 
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i 


中， 网 络 是 一 个 重要 的 构成 模块 , 例如 电子 邮件 、QQ 聊天 、 网 


上 冲浪 已 


经 充斥 了 我 们 眼球 。 作 为 智能 手机 系统 ， 在 Android 平台 上 可 以 尽情 玩 享 这 些 网 络 
本 节 的 内 容 中 ， 将 通过 几 个 典型 实例 的 实现 过 程 ， 来 详细 介绍 在 Android 中 实现 网 


基本 知识 。 
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的 大 多 数 网 页 都 是 通过 “HTTP:/W/WWW.” 的 形式 显示 的 。 在 上 其 体 应 用 中 ， 一 些 需 要 


是 通过 其 参数 传递 的 


应 用 。 在 
络 编程 的 


了 解 Web 技术 的 读者 ， 对 于 HTTP 应 该 不 会 陌生 ，HTTP 是 一 种 网 络 传输 协议 ， 生 活 中 


的 数据 都 


。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 传 


递 HTTP 参数 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\8\examplel1”， 下 面 


本 实例 的 其 体 实现 流程 。 


8.1.1 实现 原理 


台 讲 解 


和 网 络 HTTP 有 关 的 协议 是 HTTP 协议 ,在 Android SDK 中 , 集成 了 Apache 的 HttpClient 


模块 。 通 过 此 模块 ， 可 以 方便 地 编写 出 和 HTTP 有 关 的 程序 。 在 Android SDK ! 


HttpClient 4.0 版 。 


在 本 实例 中 , 使 用 


了 两 个 按钮 , 一 个 | 


于 以 Post 方式 获取 网 站 数据 , 另外 一 个 


通常 使 用 


于 以 Get 


方式 获取 数据 ， 并 以 TextView 对 象 来 显示 由 服务 器 端 返回 的 网 页 内 容 。 当 然 首 先 
HTTP 的 连接 ， 连 接 之 后 才能 获取 Web Server 返回 的 结果 。 


8.1.2 具体 实现 


本 实例 的 主 程序 文人 


F 是 examplel.java， 下 面 开始 讲解 其 具体 实现 代码 。 


1) 分 别 声明 两 个 Button 对 象 和 一 个 TextView 对 象 ， 然 后 通过 findViewByld 构 
TextView 与 Button 对 象 。 具 体 代 码 如 下 。 


public class examplel extends Activity 


{ 


/* 分 别 声明 两 个 Button 对 象 和 一 个 TextView 对 象 */ 
private Button mButton1,mButton2; 
private TextView mTextView!; 


/** Called when the activity is first created. */ 


(@Override 


得 建立 和 
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public void onCreate(Bundle savedInstanceState) 


2) 设 定 OnClickListener 来 监听 第 一 个 按钮 的 OnClick 事件 ， 首 先 声明 网 址 字符 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/# 通 过 findViewByld 构造 器 创建 TextView 与 Button 对 象 */ 
mButtonl =(Button) fmndViewById(R.id.myButtonl ); 
mButton2 =(Button) fmndViewById(R.id.myButton2); 
ImTextViewl = (TextView) findViewByld(R.id.myTextView!1); 


Ud 


立 Post 方式 联机 ， 最 后 通过 mTextView1.setText 对 象 输出 提示 字符 。 具 体 代 人 码 如 下 。 
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/# 设 定 OnClickListener 来 监听 OnClick 事件 */ 
mButtonl.setOnClickListener(new Button.OnClickListener() 


{ 


/* 窗 盖 onClick 事件 */ 
@Override 
public void onClick(View V) 


{ 


放声 明 网 址 字符 串 */ 
String uriAPI = "http:/www.dubblogs.ce:8751/Android/Test/APLPost/index.php"; 
人 # 建 立 HTTP Post 联机 方式 */ 
HttpPost httpRequest = new HttpPost(uriAP]); 
/* 
* Post 运行 传送 变量 必须 用 NameValuePair[] 数 组 存储 
List <NameValuePair> params = new ArrayList <NameValuePair>(); 


params.add(new BasicName ValuePair("str", "I am Post String")); 
try 
{ 
人 # 发 送 HTTP 请 求 */ 
httpRequest.setEntity(new UrlEncodedFormEntity(params, HTTP.UTE 8)); 
人 # 取 得 HITP 应 答 */ 
HttpResponse httpResponse = new DefaultHttpClient().execute(httpRequest); 
/# 若 状态 码 为 200*/ 
if(httpResponse.getStatusLine().getStatusCode() == 200) 
{ 
族 获 取 应 管 字符 串 */ 
String strResult = EntityUtils.toString(httpResponse.getEntity()); 
mTextViewl.setText(strResult); 


} 


else 


{ 


mTextViewl.setText("Error Response: "+httpResponse.getStatusLine().toString()); 
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catch (ClientProtocolException e) 


1 
mTextViewl.setText(e.getMessage().toString()); 
e.printStack Trace(); 
} 
catch (IOException e) 
{ 
mTextViewl.setText(e.getMessage().toString()); 
e.printStack Trace(); 
} 
catch (Exception e) 
{ 
mTextViewl.setText(e.getMessage().toString()); 
e.printStack Trace(); 
} 
} 
)); 
3) 设 定 OnClickListener 来 监听 第 二 个 按钮 的 OnClick 事件 ， 首 先 声 明 网 址 字符 串 ， 然 后 
建立 HTTP Get 联机 方式 ， 分 别 实现 发 出 HTTP 请 求 、 获 取 应 答 字 符 串 和 删除 元 余 字 符 ， 最 后 


通过 mTextViewl.setText 输出 提示 字符 。 有 具体 代码 如 下 。 


mButton2.setOnClickListener(new Button.OnClickListener() 
{ 
@Override 
public void onClick(View v) 
{ 
/# 声 明 网 址 字符 串 交 
String UriAPI= "http:/www.XXXX.cc:8751/index.php?str=IH+am+GettString"; 
/+ 建立 HTTP Get 联机 方式 */ 
HttpGet httpRequest = new HttpGet(uriAPD; 
try 
{ 
娠 发 出 HTTP 请 求 */ 
HttpResponse httpResponse = new DefaultHttpClient().execute(httpRequest); 


放 若 状态 人 码 为 200*/ 
if(httpResponse.getStatusLine().getStatusCode() == 200) 
{ 
/# 获 取 应 答 字 符 串 头 
String strResult = EntityUtils.toString(httpResponse.getEntity()); 
记 删 除 元 余 字 符 % 


strResult = eregi replace("(\r nl\r\n\n\r)","",strResult); 
mTextViewl.setText(strResult); 


} 


else 


国 国 377 


Android 开发 完全 实战 宝典 


ImTextViewl.setText("Error Response: "+httpResponse.getStatusLine().toString()); 


} 

catch (ClientProtocolException e) 

{ 
mTextViewl.setText(e.getMessage().toString()); 
e.printStack Trace(); 

} 

catch (IOException e) 

{ 
mTextViewl.setText(e.getMessage().toString()); 
e.printStack Trace(); 

y 

catch (Exception e) 

{ 
mTextViewl.setText(e.getMessage().toString()); 
e.printStack Trace(); 


1 
} 


Lr A 


4) 设 定 字符 串 替换 方法 eregi_replace(String strFrom, String strTo, String strTarget)， 蔡 换 掉 
一 些 非法 字符 。 具 体 代 码 如 下 。 


J 
public String eregi_ replace(String strFrom, String strTo, String strTarget) 


{ 


String strPattern = "(?1)"+strFrom; 


Pattern p = Pattern.compile(strPattern); 
Matcher m = p.matcher(strTarget); 
if(m.find()) 

{ 


return strTarget.replaceAll(strFrom, strTo); 


} 


else 


{ 


return strTarget; 


} 
} 


接 下 来 在 文件 AndroidManifest.xml 中 添加 对 网 络 连接 权限 ， 具 体 代码 如 下 。 


<uses-permission android:name="android.permission.INTERNET"></uses-permission> 
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执行 后 的 效果 如 图 8-1 所 示 。 分 别 单 击 图 8-1 中 的 按钮 ， 能 够 以 不 同方 式 获取 HTTP 


名 出 多 403PM 


example1 


用 POST 获取 资料 


用 GET 获 取 资 料 


图 8-1 执行 效果 
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网 上 冲浪 功能 ， 对 于 现在 的 手机 来 说 已 经 不 是 什么 难事 。 在 本 节 的 内 容 中 ， 将 通过 一 个 
具体 实例 的 实现 ， 介 绍 在 Android 中 编程 实现 网 页 浏览 的 基本 流程 。 本 实例 的 源 代码 保存 在 
“光盘 :\daima\8\example2”， 下 面 开 始 讲 解 本 实例 的 具体 实现 流程 。 


8.2.1 实现 原理 


在 Android 中 ， 内 置 了 一 个 WebKit 引擎 ， 里 面 的 WebView 组 件 能 够 迅速 实现 网 页 浏览 。 
在 本 实例 中 ,通过 WebView.loadUrl 方法 来 加 载 网 址 , 所 以 从 EditText 中 传 入 要 浏览 的 网 址 后 ， 
就 可 以 在 WebView 中 加 载 网 页 的 内 容 了 。 


8.2.2 具体 实现 


本 实例 的 主 程序 文件 是 example2.java， 下 面 开 始 讲解 其 具体 实现 代码 。 
通过 setOnClickListener 监听 按钮 单 击 事件 ， 单 击 箭头 后 先 获 取 EditText 中 的 数据 ， 然 后 
打开 此 网 址 ， 并 在 WebView 中 显示 网 页 内 容 。 有 具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


ImImageButtonl = (ImageButton)findViewByld(R.id.myImageButton!1); 


mEditTextl = (EditText)findViewById(R.id.myEditText1); 
mWebViewl = (WebView) fndViewById(R.id.myWebViewl); 
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人/# 当 单 击 箭头 后 六 


mlmageButton1.setOnClickListener(new 
ImageButton.OnClickListener() 


{ 

(QOverride 

public void onClick(View arg0) 

{ 

{ 
mImageButton1.setImageResource(R.drawable.go 2); 
/# 获 取 EditText 中 的 数据 */ 
String strURI = (mEditTextl.getText().toString()); 
族 ”WebView 显示 网 页 内 容 */ 
mWebView!l .loadUrl(strURT); 
Toast.makeText( 
example2.this,getString(R.string.load)+strURI, 
Toast. LENGTH LONG) 


.Sshow(); 


外 


执行 后 显示 一 个 文本 框 , 在 此 可 以 输入 网 址 ， 如 图 8-2 所 示 。 输入 网 址 并 单 击 后 面 的 > 
后 ， 将 显示 此 网 页 的 内 容 。 如 图 8-3 所 示 。 


[3 国名 丽 国 4:17 mm 


图 http://3g.163.corm/x/ 中 


纲 易 手机 网 易 网 
1653.COIm 
济南 [ 找 ] 闪 237C ”35T 


回 导航 ”高 档 版 莹 通 版 电脑 版 


博客 微 博 游戏 ”有 首 
[二 园 古 罗 鲍 4:15rv 新 闻 娱乐 体育 军事 下 载 


example2 股票 NBA 手机 论坛 读书 


波斯 王子 | 电影 | | 有 道 搜索 


| 


Co 一 一 一 一 


世界 杯 赛程 积分 射手 图 片 


“乌拉 圭 2-1 韩 国 40 年 后 首 进 8 强 
“ 您 如 何 看待 圭 国 队 延 憾 出 局 


图 8-2 输入 网 址 图 8-3 打开 的 网 页 
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HTML 语言 是 当前 主流 的 网 页 技术 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 
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在 Android 中 使 用 HTML 程序 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\8\example3”， 
下 面 开始 讲解 本 实例 的 具体 实现 流程 


8.3.1 ”实现 原理 


实际 上 ，WebView 是 一 个 租 入 式 的 浏览 器 ， 可 以 直接 使 用 WebView.loadData() 方 法 ， 将 
HTML 标记 传递 给 WebView 对 象 ， 让 Android 手机 程序 具备 Web 浏览 器 的 功能 。 这 样 ， 网 页 
程序 被 放 在 了 WebView 中 运行 ， 如 同一 个 Web 程序 。 在 当前 移动 程序 中 ， 网 页 下 载 和 动画 展 
示 等 都 利用 了 WebView 中 的 loadData0) 方 法 来 载 入 网 页 。 


8.3.2 具体 实现 


本 实例 的 主 程序 文件 是 example3.java， 在 loadData() 方 法 中 插入 了 指定 的 HTML 代码 ， 
过 HTML 代码 ， 显 示 了 一 幅 图 片 和 文字 ， 并 且 插 入 了 超级 链接 功能 。 具 体 代码 如 下 。 


o 


EE) 


Ea 


package irdc.example3; 


import irdc.example3.R; 

import android.app.Activity; 
import android.os.Bundle; 

import android.webkit. WebView; 


public class example3 extends Activity 
{ 
private WebView mWebView!; 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mWebViewl = (WebView) findViewById(R.id.myWebView!1); 


人 # 自 行 设置 WebView 要 显示 的 网 页 内 容 */ 
mWebView1. 
loadData( 


"<html><body><p>aaaaaaa</p>" 十 


"<div class="widget-content'> "+ 

"<a href=http://www.sohu.com>" 十 

"<img src=http://hiphotos.baidu.com/chaojihedan/pic/itenm/bbddfSefc260f133fdfa3cd8.jpg />"+ 
"<a href=http://www.sohu.com>Link Blog</a>" + 

"</body></html>", "text/html", "utf-8"); 


} 


执行 后 将 显示 HTML 产生 的 页 面 ， 如 图 8-4 所 示 。 单 击 超 链 接 后 会 跳 转 到 指定 的 目标 页 
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面 。 如 图 8-5 所 示 。 


星 5554: ff 


PM 


example3 


图 http://www.sohu.c... 闻 X 


图 8-4 输入 网 址 图 8-5 打开 的 网 页 


过 


前 面 儿 个 实例 实际 上 是 Android 浏览 器 的 部 分 功能 ， 实 际 上 用 户 可 以 直接 调用 它 的 内 置 
浏览 器 ， 来 实现 上 网 操作 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 调用 Android 
内 置 浏 览 器 的 基本 流程 。 本 实例 的 源 代 码 保 存在 “光盘 :daimaNg\example4”， 下 面 开 始 讲解 本 
实例 的 具体 实现 流程 。 


8.4.1 实现 原理 
在 本 实例 中 ， 定 义 了 一 个 ListView， 列 表 显 示 了 4 个 菜单 ， 单 击 染 单 后 会 连接 到 指定 的 
页 面 。 当 ListView 的 ItemClick0 事 件 发 生 时 ， 通 过 Intent(mntentACTION VIEWuri) 方 法 来 打 
开 内 置 的 浏览 器 ， 并 浏览 ListView 中 创建 的 网 页 URL 
8.4.2 ”具体 实现 
的 主 程序 文件 是 example4.java， 下 面 开 始 介绍 其 实现 流程 。 


分 别 声明 一 个 ListView 和 TextView 对 象 变 量 ， 然后 声 声明 一 个 String array 变量 来 存储 
ee 最 后 声明 一 个 String 变量 来 存储 网 址 。 具 体 代码 如 下 。 


package irdc.example4; 


public class example4 extends Activity 


{ 
/* 声 明 一 个 ListView,TextView 对 象 变量 
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2) 先 通过 findViewByld 构造 器 创建 ListView 与 TextView 对 象 ， 将 string.xml 文件 


* 一 个 String array 变量 存储 收藏 夹 列 表 
* 与 String 变量 来 存储 网 址 */ 
private ListView mListView1l; 


private TextView mTextView1l; 


private String[] myFavor; 
private String myUrl; 


言 恩 导入 到 列表 中 。 


具体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 
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/# 通 过 findViewById 构造 器 创建 ListView 与 TextView 对 象 */ 
mListViewl =(ListView) findViewById(R.id.myListViewl); 
ImTextViewl = (TextView) fndViewById(R.idmyTextViewl); 
mTextViewl.setText(getResources().getString(R.string.hello)); 


局 将 string.xml 文件 中 信息 导入 到 列表 */ 


myFavor =new String[] { 


getResources().getString 
(R.string.str_list_url1), 
getResources().getString 
(R.string.str_list_url2), 
getResources().getString 
(R.string.str_list_url3), 
getResources().getString 


(R.string.str_list_url4) 


3) 自 定义 ArrayAdapter 对 象 ， 时 刻 准 备 传 入 ListView 中 ， 并 将 myFavor 对 象 的 列表 以 


参数 传 入 。 然 后 自 定义 完成 的 ArrayAdapter 传 入 自 定 义 的 ListView 中 ， 
后 设置 ListView 选项 的 nItemClickListener。 上 有 具体 代码 如 下 。 


选 (Focusable) 菜 单 选项 打开 ， 最 


Ws 


目 定 义 一 ArrayAdapter7 


E 备 传 入 ListView 中 ， 


ArrayAdapter<String> adapter = new 


ArrayAdapter<String> 


并 将 ListAdapter 的 可 


将 myFavor 列表 以 参数 传 入 */ 


(example4 .this, android.R.layout.simple list item 1, myFavor); 


/* 将 自 


定义 完成 的 ArrayAdapter 传 入 自 定义 的 ListView 9 
mListView1.SetAdapter(adapter); 


/* 将 ListAdapter 的 可 选 (Focusable) 菜 单 选项 打开 */ 
ImListViewl.setItemsCanFocus(true); 


A 


ListView 菜单 选项 设 为 每 次 只 


能 单一 选项 % 


px 
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mListViewl.setChoiceMode 
(ListView.CHOICE MODE SINGLE); 

/* 设 置 ListView 选项 的 nItemClickListener*/ 
mListViewl.setOnItemClickListener 
(new ListView.OnItemClickListener() 


{ 


4) 定义 覆盖 onItemClick0 方 法 ， 当 用 户 单 击 一 个 Item 条目 ) 后 ， 会 进行 比较 ， 并 从 文 
件 string.xml 中 取出 对 应 的 URL 网 址 ， 并 将 字符 串 转 换 为 URL 对 象 。 具 体 代 码 如 下 。 
(@Override 
/* 禾 六 OnItemClick() 方 法 */ 
public void onItemClick 


(AdapterView<?> arg0, View argl, int arg2,long arg3) 
{ 

访 如 果 所 选 菜 单 的 文字 与 myFavor 字符 捉 数组 第 一 个 文字 相同 */ 

if(arg0.getAdapter().getItem(arg2).toString()== 

myFavor[0].toString()) 

{ 
/# 取 得 网 址 并 调用 goToUr10 方 法 */ 
ImyUrl=getResources(.getString(R.string.str_ urll); 
goToUrl(myUr)); 

} 

此 如 果 所 选 菜 单 的 文字 与 myFavor 字符 串 数组 第 二 个 文字 相同 */ 

else if (arg0.getAdapter().getltem(arg2).toString()== 

myFavor[1].toString()) 

{ 
/# 取 得 网 址 并 调用 goToUr10 方 法 */ 
myUrl=getResources().getString(R.string.str_url2); 
goToUrl(myUr)}); 

} 

人 # 如 果 所 选 菜单 的 文字 与 myFavor 字符 串 数 组 第 三 个 文字 相同 */ 

else if (arg0.getAdapter().getltem(arg2).toString()== 


myFavor[2].toString()) 

{ 
/# 取 得 网 址 并 调用 goToUrl0 方 法 */ 
myUrl=getResources().getString(R.string.str_url3); 
goToUrl(myUr)}); 

} 

/# 如 果 所 选 菜单 的 文字 与 myFavor 字符 串 数 组 多 

else if (arg0.getAdapter().getltem(arg2).toString()== 

myFavor[3].toString()) 

{ 
/# 取 得 网 址 并 调用 goToUr10 方 法 */ 
myUrl=getResources().getString(R.string.str_url4); 
goToUrl(myUr)); 

} 


瀑 


I 个 文字 相同 刀 
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DL EEY 

else 

{ 
放 显 示 错误 信息 */ 
mTextViewl.setText("Ooops!! 出 错 了 "); 


} 


») 
} 


5) 定义 方法 goToUrl(String url))， 用 于 打开 网 址 为 URL 的 网 页 。 有 具体 代码 如 下 。 


人 # 打 开 网 页 的 方法 光 

private void goToUrl(String url) 

{ 
Uri uri = Uri.parse(ur}); 
Intent intent = new Intent(Intent.ACTION VIEW, uri); 
startActivity(intent); 


} 


执行 后 将 列表 显示 4 个 表单 ， 如 图 8-6 所 示 。 当 单 击 一 个 菜单 后 ， 会 跳 转 到 对 应 的 目标 


由 H|。 如 图 8-7 所 示 。 


example4 


163 


另 而 丘 4:27 PM 


园 httpy/www.baidu.com/ 只 


Bui 轨 mm 


baidu 
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图 8-6 4 个 菜单 图 8-7 打开 的 网 页 
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网 络 真是 很 神奇 ， 在 QQ 空 


闻 中 可 以 存放 自己 的 照片 。 我 们 有 时 并 不 需要 在 Gallery 中 


存放 照片 ， 可 以 直接 从 网 络 中 调用 照片 ， 并 在 Gallery 中 显示 出 来 ， 这 样 可 以 节约 手机 的 存 
储 空 间 。 在 本 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 Gallery 中 调用 网 络 照片 显示 


的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\8\example5”， 下面 开始 讲解 本 实例 的 县 


体 实现 流程 。 


8.5.1 实现 原理 


~ 


在 本 实例 中 ， 需 要 将 URL 网 址 的 照片 实时 处 理 下 载 后 ， 以 InputStream 输 入 流 ) 转换 为 
Bitmap 图 像 ， 这 样 才能 放 入 BaseAdapter 中 。 在 运行 实例 前 ， 需 要 预先 准备 照片 
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络 空间 中 , 获取 照片 的 连接 后 , 再 以 String 数组 方式 放 在 程序 中 , 并 对 BaseAdapter 稍 作 修改 ， 
增加 对 URL 对 象 的 访问 以 及 对 URLConnection 连接 的 处 理 。 


8.5.2 ”具体 实现 
本 实例 的 主 程序 文件 是 example5.java， 下 面 开 始 介 绍 其 实现 流程 。 
1) 分 别 声明 Gallery 中 要 显示 5 幅 图 片 的 地 址 栏 字符 串 ， 具 体 代码 如 下 。 


T 


public class exampleS extends Activity 


{ 
private Gallery myGallery01; 
入 地 址 栏 字符 串 
private String[] myImageURL = new String[] 
{ 


"http://b27.photo.store.qq.com/http _ imgload.cgi?/" 
十 
"rurl4 b=086a67cbd6a8cfb4389ea2b48efab6f322f755a085107a7aeeaaS6fc1358blbd124186254e021f0655732688e6 
9f060725491f8ae82e8e5508dbe9821670e2baf04e92dedc97e3bbf28e5605596aa991c13220f1 &a=27&b=27", 
"http://b27.photo.store.qq.com/http_ imgload.cgi?/" 
十 
"rurl4 b=086a67cbd6a8cfb4389ea2b48efab6f3ea78f5797abbbaa617259f2d2a980a5468f2801897cfcc2b78af92fbb87 
565ed7a3a08041daff2dd9ccd26d3cc6198e41f2d205c8a0c445325771e8al79215999afaf9f3&a=27&b=27", 
"http://b27.photo.store.qq.com/http_ imgload.cgi?/" 
十 
"rurl4 b=2a9dcflfd909a7ed3ce8951f738608982f26d812b3aSfc96e221b85fc085e7cc3264ee20730f0fd3alf7aca0674 
0db7a6153d9357467ca39f82b866b6fbe3cd94bbdd10ed01841e67c95d8e4af8890b7ced40869&a=30&b=27", 
"http://b27.photo.store.qq.com/http imgload.cgi?/" 
十 
"rurl4 b=2a9dcflfd909a7ed3ce8951f73860898bb7ffs7a8cb7747c9f0eb6a02124850b709c0b86f086a4ba5653eeb71d 
d4b01le4a58f407e2eec9433cd8d4bc0b88fda56260c2c8beb34ebab77b610c7131393f82e774ef&a=27&b=27", 
"http://b27.photo.store.qq.com/http_ imgload.cgi?/" 
十 
"rurl4 b=2a9dcflfd909a7ed3ce8951f73860898158d252489f84e7d2a83d44c01b7bb12b2c19ca0efdd555dba788407fd 
Ole9de45524b11a9793f532624197bc8d14c84ae78ddebafe4357e4eedc60e9e510224367490bf&a=27&b=27" }; 


2) 引入 布局 文件 main.xml， 定 义 类 成 员 myContext Context 对 象 ， 然 后 设置 只 有 一 个 参 
数 “C” 的 构造 器 ， 即 要 存储 的 Context。 获 取 Gallery 属性 的 Index id， 并 设置 对 象 的 styleable 
属性 ， 使 其 能 够 反复 使 用 。 具 体 代码 如 下 。 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


myGallery01 = (Gallery) findViewById(R.id.myGallery01); 
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方法 ， 使 用 getltem 方法 获取 当前 
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;去 
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myGallery01.setAdapter(new myInternetGalleryAdapter(this)); 
} 


/* 用 BaseAdapter */ 
public class myInternetGalleryAdapter extends BaseAdapter 
{ 

雍 类 成 员 myContext Context 对 象 */ 

private Context myContext; 

private int mGalleryItemBackground; 


放 构 造 器 只 有 一 个 参数 ， 即 要 存储 的 Context */ 
public myInternetGalleryAdapter(Context c) 
{ 


this.myContext = ¢; 


TypedArray a = myContext 
.obtainStyledAttributes(R.styleable.Gallery); 


/* ”获取 Gallery 属性 的 Index id */ 
mGalleryItemBackground = a.getResourceld( 
R.styleable.Gallery android galleryItemBackground, 0); 


/* 把 对 象 的 styleable 属性 能 够 反复 使 用 */ 
a.recycle(); 


} 


3) 定义 方法 getCount0， 用 于 返回 已 定义 图 片 的 总 量 。 然 后 定义 方法 getItem(int position) 


> 


器 中 图 像 数 的 数组 ID 。 具 体 代 码 如 下 。 


人 # 返回 全 部 已 定义 图 片 的 总 量 六 
public int getCount() 


{ 
return myImageURL.length; 


} 


旋 使 用 getttem 方法 获取 当前 容器 中 图 像 数 的 数组 ID */ 
public Object getItem(int position) 
{ 

return position; 


} 


public long getItemld(int position) 
{ 


return position; 


} 


4) 定义 方法 getScale(boolean focused, int offset) 方 法 ， 根 据 中 央 位 移 量 ， 利 用 getScale0 


返 


回 | views 的 大 小 (0.0fto 1.0f。 具 体 代码 如 下 。 
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必 根据 中 央 位 移 量 ， 利 用 getScale 返回 views 的 大 小 (0.0fto 1.0f) */ 


public float getScale(boolean focused, int offset) 
{ 
/* Formula: 1/ (2 offset) */ 
return Math.max(0, 1.0f/ (float) Math.pow(2, Math 
.abs(offset))); 
} 


5) 定义 getView0 方 法 ,根据 中 央 位 移 量 ， 获 取 当 前 要 显示 


的 图 像 View， 传 入 数组 ID 值 


使 之 读 取 并 成 像 处 理 。 流 程 如 下 。 
第 一 步 : 创建 ImageView 对 象 。 
第 二 步 : 用 new URL 将 对 象 网 址 传 入 。 
第 三 步 : 获取 连接 。 
第 四 步 : 获取 返回 的 输入 流 。 
第 五 步 : 将 InputStream 变 为 Bitmap。 
第 六 步 : 关闭 InputStream 。 

第 七 步 : 设置 Bitmap 到 ImageView 中 。 


也 


第 八 步 : 设置 ImageView 的 宽 和 高 ， 单 位 是 dip。 


er tl 


第 九 步 : 设置 Gallery 背景 图 。 
生体 代码 如 下 


(@Override 
public View getView(int position, View convertView, 


ViewGroup parent) 


{ 
访 创建 ImageView 对 象 */ 


ImageView imageView = new ImageView(this.myContext); 
try 
{ 
/* new URL 将 对 象 网 址 传 入 */ 
URL aryURI= new URL(myImageURL[position)]); 
上 # 获取 连接 * 
URLConnection conn = aryURI.openConnection(); 
conn.connect(); 
/# 获取 返回 的 InputStream */ 
InputStream is = conn.getInputStream(); 
/* 将 InputStream 变 为 Bitmap */ 
Bitmap bm = BitmapFactory.decodeStream(is); 
/# 关闭 mputStream */ 
is.close(); 
/* 设置 Bitmap 到 ImageView 中 */ 
imageView.setImageBitmap(bm); 
} catch (IOException e) 
{ 
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e.printStackTrace(); 


} 


imageView.setScaleType(ImageView.ScaleType.FIT XY); 

/# 设置 ImageView 的 宽 和 高 ， 单 位 是 dip */ 
imageView.setLayoutParams(new Gallery.LayoutParams(200, 150)); 
/* 设置 Gallery 背景 图 */ 
imageView.setBackgroundResource(mGalleryIltemBackground); 
return imageView; 


} 
} 


执行 后 将 在 Gallery 中 显示 网 络 中 的 图 片 。 如 图 8-8 所 示 。 


图 8-8 执行 效果 


8.6 00D0D0 MP3 


同样 是 为 了 节约 手机 的 存储 空间 ， 在 听 音 乐 时 可 以 从 网 络 中 下 载 播放 MP3。 在 本 节 的 内 
容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 网 络 下 载 播放 MP3 的 基本 流程 。 本 实 
例 的 源 代 码 保存 在 “光盘 :daimaNg\example6”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


8.6.1 实现 原理 

在 本 实例 中 ， 首 先 插入 4 个 按钮 ， 分 别 用 于 播放 、 和 暂停 、 重 新 播放 和 停止 处 理 。 执 行 后 ， 

通过 Runnable 接口 发 起 运行 线程 ， 在 线程 中 远程 下 载 指定 的 MP3 文件 。 下 载 完毕 后 ， 临 时 

保存 到 SD 卡 中 ， 这 样 可 以 通过 4 个 按钮 对 其 进行 控制 。 当 程序 关闭 后 ， 删 除 SD 卡 中 的 临时 
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性 文件 。 
8.6.2 ”具体 实现 
本 实例 的 主 程序 文件 是 example6.java， 下 面 开 始 介绍 其 实现 流程 


1) 先 加 载 相 关 类 ， 然 后 声明 系统 中 需要 的 私有 变量 ， 有 具体 代码 如 下 。 


public class example6 extends Activity 
{ 
private TextView mTextView01; 
private MediaPlayer mMediaPlayer01; 
private ImageButton mPlay, mReset, mPause, mStop; 
private boolean bISReleased = false; 
private boolean bIsPaused = false; 
private static final String TAG = "Hippo_ URL MP3 Player"; 


》 


2) 定义 currentFilePath 用 于 记录 当前 正在 播放 MP3 的 地 址 ， 定 义 currentTempFilePath 变 


| 


生 


示 当 


前 播放 MP3 的 路 径 。 具 体 代码 如 下 。 
谨 记录 当前 正在 播放 MP3 的 地 址 忆 


private String currentFilePath = ""; 


/* 当 前 播放 MP3 的 路 径 */ 
private String currentTempFilePath = ""; 
private String strVideoURL = ""; 


和 


3) 图 


E 引 入 主 布局 文件 main.xml， 然 后 通过 strVideoURL 对 象 设置 要 播放 MP3 文件 的 网 


/** Called when the activity is first created. */ 


(QOverride 


public void onCreate(Bundle savedInstanceState) 
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{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


从 mp3 文件 不 会 被 下 载 到 本 地 */ 
strVideoURL = "http://www.lrn.cn/zywh/xyyy/yyXs/200805/WO20080505536315331317.MP3"; 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 


人 设置 透明 度 沁 
getWindow().setFormat(PixelFormat.TRANSPARENT); 


ImPlay = (ImageButton)findViewById(R.id.play); 
mReset = (ImageButton)findViewById(R.id.reset); 
mpPause = (ImageButton)findViewById(R.id.pause); 
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mStop = (ImageButton)findViewById(R.id.stop); 
4) 设置 单 击 “ 播 放 ” 按 钮 所 触发 的 处 理事 件 ， 具 体 代 码 如 下 。 


入 播放 按钮 */ 
mpPlay.setOnClickListener(new ImageButton.OnClickListener() 
{ 
public void onClick(View view) 
{ 
入 调用 播放 影片 函数 */ 
playVideo(strVideoURL); 
mTextView01.setText 
( 
getResources().getText(R.string.str_ play).toString()+ 
"\n"+ strVideoURL 
); 
} 
)); 


5) 设置 单 击 “ 重 新 播放 ”按钮 所 触发 的 处 理事 件 ， 具 体 代码 如 下 。 


人 # 重新 播放 */ 
mReset.setOnClickListener(new ImageButton.OnClickListener() 
{ 
public void onClick(View view) 
{ 
if(bIsReleased 一 false) 
{ 
if (mMediaPlayer01 != null) 
{ 
mMediaPlayer01.seekTo(0); 
mTextView01.setText(R.string.str_play); 
} 


} 
人 


6) 设置 单 击 “ 和 暂停 播放 ”按钮 所 触发 的 处 理事 件 ， 具 体 代码 如 下 。 
和 泡 


mPause.setOnClickListener(new ImageButton.OnClickListener() 


{ 


public void onClick(View view) 


{ 
if (mMediaPlayer01 != null) 
{ 
if(bIsReleased 一 false) 
{ 


二 
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if(bIsPaused==false) 
{ 
mMediaPlayer01.pause(); 
bIsPaused = true; 
mTextView01.setText(R.string.str pause); 
} 
else 1f(bIsPaused==true) 
{ 
mMediaPlayer01.start(); 
bIsPaused = false; 
mTlextView01.setText(R.string.str play); 
} 


} 


7) 设置 单 击 “ 停 止 播放 ”按钮 所 触发 的 处 理事 件 ， 


/终止 */ 


具体 代码 如 下 。 


mStop.setOnClickListener(new ImageButton.OnClickListener() 


{ 
public void onClick(View view) 
{ 
try 
{ 
if (mMediaPlayer01 != null) 
{ 
if(bIsReleased==false) 
{ 
ImMediaPlayer01.seekTo(0); 
mMediaPlayer01.pause(); 
//mMediaPlayer01 .stop(); 
//mMediaPlayer01 .release(); 
//bIsReleased = true; 
mTextView01.setText(R.string.str_stop); 


} 

catch(Exception e) 

{ 
mTextView01.setText(e.toString()); 
Log.e(TAG, e.toString()); 
e.printStack Trace(); 
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8) 定义 方法 playVideo(final String strPath)， 用 于 播放 指定 的 MP3， 其 播放 的 是 存储 卡 中 


暂时 保存 的 MP3 文件 ， 有 具体 代码 如 下 。 


private void playVideo(final String strPath) 
{ 
try 
{ 
if (strPath.equals(currentFilePath)&& mMediaPlayer01 != null) 
{ 
mMediaPlayer01 .start(); 


return; 


currentFilePath = strPath; 


mMediaPlayer01 = new MediaPlayer(); 
mMediaPlayer01.setAudioStreamType(2); 


9) 定义 setOnErrorListener 实现 错误 处 理 ， 有 具体 代码 如 下 。 


局 错误 事件 % 
ImMediaPlayer01.setOnErrorListener(new MediaPlayer.OnErrorListener() 
{ 
Override 
public boolean onError(MediaPlayer mp, int what, int extra) 
{ 
/TODO Auto-generated method stub 
Log.i(TAG, "Error on Listener, what: " + what + "extra: " + extra); 


return false; 


} 
号 


件 。 上 基体 


oy 


10) 定义 setOnBufferingUpdateListener， 捕 捉 使 用 MediaPlayer 缓冲 区 的 更 新 
代码 如 下 。 


诺 捕捉 使 用 MediaPlayer 缓冲 区 的 更 新 事件 */ 
mMediaPlayer01.setOnBufferingUpdateListener(new MediaPlayer.OnBufferingUpdateListener() 
{ 

(QOverride 

public void onBufferingUpdate(MediaPlayer mp, int percent) 

{ 

Log.i(TAG, "Update buffer: " + Integer.toString(percent)+ "%"); 
} 


贡品 
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11) 定义 setOnCompletionListener， 实 现 播 放 完 毕 所 触发 的 事 伯 


12) 定义 setOnPreparedListener， 


让 


具体 代码 如 下 。 


o 
| 


上 播放 完毕 所 触发 的 事件 */ 
mMediaPlayer01.setOnCompletionListener(new MediaPlayer.OnCompletionListener() 


{ 


Override 
public void onCompletion(MediaPlayer mp) 


{ 


delFile(currentTempFilePath); 
Log.i(TAG,"mMediaPlayer01 Listener Completed"); 


) 
人 


了 于 开始 阶段 的 监听 Listener。 上 有 具 体 代 码 如 下 。 


Wa 


谍 开始 阶段 的 监听 Listener */ 
mMediaPlayer01.setOnPreparedListener(new MediaPlayer.OnPreparedListener() 


{ 


Override 
public void onPrepared(MediaPlayer mp) 


{ 


Log.i(TAG,"Prepared Listener"); 


lh 
4 


13) 定义 Runnable 对 象 f， 


将 文件 存 到 SD | 


用 Runnable 来 确保 文件 在 存储 完毕 后 才 开始 start0 方 法 。 先 


> 


上 E， 然 后 在 运行 setDataSource() 方 法 后 运行 prepare() 方 法 ， 最 后 通过 mMedia 


Player01.start() 方 法 开始 播放 MP3。 具 体 代码 如 下 。 
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入 用 Runnable 来 确保 文人 


在 存 侍 完 毕 后 才 始 start() 2/ 


Runnabler = new Runnable() 


{ 


public void run() 


{ 
try 
{ 


/* setDataSource 将 文件 存 到 SD 卡 */ 
setDataSource(strPath); 


A 


为 为 线程 顺利 ; 


行 ， 所 以 在 setDataSource 后 运行 prepare() */ 


mMediaPlayer01.prepare(); 
Log.i(TAG, "Duration: " + mMediaPlayer01.getDuration()); 


WW 


始 播 放 mp3 */ 


mMediaPlayer01.start(); 
bIsReleased = false; 


第 8 章 _ 网 络 应 用 “了 


} 
catch (Exception e) 
{ 
Log.e(TAG, e.getMessage(), e); 
} 
} 
}; 
new Thread(r).start(); 
} 


14) 有 有 异常 则 输出 提示 ， 上 其 体 代码 如 下 。 


catch(Exception e) 
{ 
if (mMediaPlayer01 != null) 
{ 
雍 线程 发 生 异 常 则 停止 播放 ” */ 
ImMediaPlayer01.stop(); 
mMediaPlayer01.release(); 
) 
e.printStackTrace(); 
} 
} 


15) 定义 方法 setDataSource()， 用 于 存储 URL 的 MP3 文件 到 存储 卡 。 首 先 判 断 传 入 的 地 
址 是 否 为 URL， 然 后 创建 URL 对 象 和 临时 文件 ， 当 fos 存储 完毕 调用 MediaPlayer setData 


Source() 方 法 。 有 具体 代码 如 下 。 


/* ”定义 方法 用 于 存储 URL 的 MP3 文件 到 存储 卡 ”*/ 
private void setDataSource(String strPath) throws Exception 
{ 

谍 ”判断 传 入 的 地 址 是 否 为 URL */ 

if (IURLUtil.isNetworkUrl(strPath)) 


{ 
mMediaPlayer01.setDataSource(strPath); 
} 
else 
{ 
if(bIsReleased — false) 
{ 


/* 创建 URL 对 象 */ 

URL myURL = new URL(strPath); 
URLConnection conn =myURL.openConnection(); 
conn.connect(); 


/x ”获取 URL 地 址 的 输入 流 */ 


InputStream is = conn.getInputStream(); 
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if (is == null) 
{ 


throw new RuntimeException("stream is null"); 


/# 创建 临时 文件 六 


File myTempFile = File.create TempFile("yinyue", "."+getFileExtension(strPath)); 


currentTempFilePath = myTempFile.getAbsolutePath(); 


/* currentTempFilePath = /sdcard/hippoplayert MP39327.MP3 */ 


/* 
if(currentTempFilePath!="") 
{ 
Log.i(TAG, currentTempFilePath); 
} 
人 


FileOutputStream fos = new FileOutputStream(myTempFile); 
byte buf[] = new byte[128]; 
do 
{ 

int numread = is.read(bu?f); 

if (numread <= 0) 

{ 

break; 

} 

fos.write(buf, 0, numread); 
}while (true); 


/* 直到 fos 对 象 存储 完毕 ， 调 用 MediaPlayer.setDataSource */ 
mMediaPlayer01.setDataSource(currentTempFilePath); 

try 

{ 


is.close(); 


} 


catch (Exception ex) 


{ 
Log.e(TAG, "error: " + ex.getMessage(), ex); 


} 


16) 定义 方法 getFileExtension(String strFileName)， 获 取 音 乐 文件 扩 
获取 扩展 名 ， 则 默认 为 .dat。 有 具体 代码 如 下 。 
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展 名 ， 如 果 无 法 顺利 


放 ”获取 音乐 文件 扩展 名 自 定义 方法 * 
private String getFileExtension(String strFileName) 


{ 


File myFile = new File(strFileName); 


String strFileExtension=myFile.getName(); 


下 


草 
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strFileExtension=(strFileExtension.substring(strFileExtension.lastIndexOf(".")+1)).toLowerCase(); 


if(strFileExtension=="") 
{ 
旋 如 果 无 法 顺利 获取 扩 


strFileExtension = "dat"; 


展 名 则 默认 为 .dat */ 


} 


return strFileExtension; 


} 


【LE 


[ 守 


17) 定义 方法 delFile(String strFileName)， 当 


下 所 示 : 


开 程序 时 删除 临时 音乐 文件 


BT 
FE 


7 程序 时 需要 调 
private void delFile(String strFileName) 
{ 
File myFile = new File(strFileName); 
if(myFile.exists()) 
{ 
myFile.delete(); 


(@Override 
protected void onPause() 


{ 


J 
try 


{ 
delFile(currentTempFilePath); 


} 


catch(Exception e) 


{ 


e.printStack Trace(); 


} 


super.onPause(); 


删除 临时 文件 六 


} 
执行 后 可 以 通过 所 


放 、 和 暂停 


、 重 新 播放 和 停止 四 个 按钮 ， 控 


用 上 自 定义 方法 删除 临时 音乐 文件 


十 
日 


曾 指定 MP3 文人 


具体 代码 如 
F 的 处 理 。 效 
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果 如 图 8-9 所 示 。 


ff SEE 


example6 


上 在 播放 
http:wwww.Irn.cn/zywhy/xyyy/yyxs/200805/ 
020080505536315331317.mp3 


四 


本 | | a EE Ee 


图 8-9 ”执行 效果 
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铃声 是 移动 手机 的 重要 功能 之 一 ， 同 样 也 可 以 从 网 络 中 直接 下 载 一 个 MP3 文件 ， 并 设置 
为 手机 的 铃声 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 远程 下 载 
手机 铃声 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\8\example7”， 下面 开始 讲解 本 实 
例 的 具体 实现 流程 。 


8.7.1 实现 原理 

在 本 实例 中 ， 用 户 通 过 EditText 控件 输入 一 个 指定 的 网 址 ， 当 下 载 完 成 后 ， 打 开 
RingtoneManagerACTION RINGTONE PICKER 这 个 mtent， 在 打开 Intent 的 同时 传 入 一 个 参数 ， 
这 个 ACTION RINGTONE PICKER 的 Intent 会 带 入 刚才 下 载 文件 让 用 户 选 择 。 在 实现 过 程 中 ， 
会 判断 下 载 文件 是 否 完整 ， 并 判断 用 户 已 设置 铃声 文件 、 下 载 后 的 铃声 文件 存储 在 哪里 。 在 具体 
实现 时 , 会 以 SD 卡 中 的 铃声 文件 路 径 作为 存储 网 络 下 载 音 乐 文件 的 路 径 , 打开 RingtoneManager 
的 ACTION RINGTONE PICKER 的 Intent， 让 用 户 找到 下 载 的 音乐 ， 并 作为 铃声 。 


8.7.2 ”具体 实现 
本 实例 的 主 程序 文件 是 example7.java， 下 面 开始 介绍 其 实现 流程 。 


1) 先 引 入 相关 类 ， 然 后 声明 系统 中 需要 的 对 象 ， 具 体 代码 如 下 。 


public class example7 extends Activity 


{ 
protected static final String APP TAG = "DOWNLOAD RINGTONE"; 
private Button mButton1; 
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private TextView mTextView1l; 
private EditText mEditTextl; 
private String strURL = ""; 

public static final int RINGTONE PICKED = 0x108; 


= 


private String currentFilePath 
private String currentTempFilePath = "",; 
private String fileEx=""; 
private String fileNa=""; 


private String strRingtoneFolder = "/sdcard/music/ling"; 


2) 判断 是 否 有 /sdcard/music/ringtones 文件 夹 ， 不 存在 则 输出 提示 。 具 体 代 码 如 下 。 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


mButtonl =(Button) findViewById(R.id.myButton!); 
ImTextViewl = (TextView) fndViewById(R.id.myTextViewl); 
mEditTextl = (EditText) fndViewById(R.id.myEditTextl); 


访 判 断 是 否 有 /sdcard/music/ringtones 文件 夹 */ 
if(bIfExistRingtoneFolder(strRingtoneFolder)) 
{ 

Log.i(APP TAG, "Ringtone Folder exists."); 


} 


3) strURL 是 在 String 中 设置 的 , 通过 fileEx0 方 法 和 getFile0 方 法 取得 文件 的 名 称 。 具体 
代码 如 下 。 


mButtonl.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View arg0) 


{ 
strURL = mEditTextl.getText().toString(); 


Toast.makeText(example7 .this, getString(R.string.str_ msg) 
,Toast.LENGTH SHORT).show(); 


放 取 得 文件 名 称 */ 

fileEx = strURL.substring(strURL.lastIndexOf(".")+1,strURL. 
length()).toLowerCase(); 

fileNa = strURL.substring(strURL.lastIndexOf("/")+1,strURL. 
lastIndexOf(".")); 
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} 


4) 定义 方法 getMIMEType(File ff， 判断 文件 类 型 的 方法 
展 名 的 类 型 决定 MimeType。 上 基体 代 码 如 下 。 


400O 看 蛋 


getFile(strURL); 


} 
2 


上 六 判断 文件 MimeType 的 method */ 


private String get MIMEType(File f) 


{ 
String type=""; 
String fName=f.getName(); 
人 取得 扩展 名 */ 
String end=fName.substring(fName.lastIndexOf(".")+1, 
fName.length()).toLowerCase(); 


上 # 依 扩展 名 的 类 型 决定 MimeType */ 
ji 放 end.equals("m4a")llend.equals("MP3")llend.equals("mid" 
end.equals("xmf"')llend.equals("ogg")llend.equals("wav")) 


{ 
type = "audio"; 
} 
else if(end.equals("3gp")llend.equals("mp4")) 
{ 
type = "video"; 
} 


else if(end.equals("jpg")llend.equals("gif")|| 
end.equals("png")llend.equals("jpeg") 川 
end.equals("bmp")) 


{ 
type = "image"; 
} 
else 
{ 
type="*"; 
} 


= 如 果 无 法 直接 打开 ， 就 跳出 软件 列表 给 用 户 选择 */ 
if(end.equals("image")) 
{ 
} 
else 
{ 
type += "/*"; 


} 
return type; 


。 首 先 取得 扩展 名 ， 然 后 根据 扩 


第 8 章 网 络 应 用 


5) 定义 方法 getFile(final String strPath)， 
则 直接 


We 


private void getFile(final String strPath) 
{ 
try 


{ 
if (strPath.equals(currentFilePath) ) 


{ 


getDataSource(strPath); 


} 


currentFilePath = strPath; 
Runnabler = new Runnable() 


{ 
public void run() 


{ 
try 
{ 


getDataSource(strPath); 


} 


catch (Exception e) 


{ 
Log.e(APP TAG, e.getMessage(), e); 
} 
} 
}; 
new Thread(r).start(); 


} 


catch(Exception e) 


{ 


e.printStack Trace(); 


} 
} 


6) 定义 方法 getDataSource(String strPath)， 
误 信 息 。 流 程 如 下 。 
口 第 一 步 : 通过 myURL 获取 URL。 
第 二 步 : 创建 连接 conn。 
第 三 步 : 通过 InputStream 下 载 文件 。 
第 四 步 : 创建 文件 地 址 myTempFile。 
第 五 步 : 取得 在 内 存 中 的 存储 路 径 。 
第 六 步 : 将 文件 写 入 和 暂 存盘 。 
L 体 代码 如 下 。 


从 取得 远程 文件 * 


DODODD 


es 


EE 


用 于 获 
j getDataSource0 方 法 数据 。 如 果 有 异常 ， 则 输出 异常 信息 。 具 体 代码 如 下 。 


取 最 后 的 文件 。 如 果 地 址 和 当前 地 址 一 样 ， 


于 获取 远程 文件 。 如 


Se 


四 
个 


地 址 错误 ， 则 输 错 
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private void getDataSource(String strPath) throws Exception 


{ 


if (IURLUtil.isNetworkUrl(strPath)) 
{ 
mTextViewl.setText(" 错 误 的 URL"); 
} 
else 
{ 
入 取得 URL*/ 
URL myURL = new URL(strPath); 
放 创 建 连接 
URLConnection conn = myURL.openConnection(); 


conn.connect(); 
/*InputStream 下 载 文 件 */ 
InputStream is = conn.getInputStream(); 
if (is == null) 
{ 
throw new RuntimeException("stream is null"); 


} 


娠 创建 文件 地 址 党 
File myTempFile = new File("/sdcard/music/ling/", 
fileNat"."+fileEx); 

族 取 得 在 内 存 中 的 存储 路 径 */ 
currentTempFilePath = myTempFile.getAbsolutePath(); 
/将 文件 写 入 暂 存 稳 沁 
FileOutputStream fos = new FileOutputStream(myTempFile); 
byte buf[] = new byte[128]; 
do 
{ 

int numread = is.read(buf); 

if (numread <= 0) 

{ 

break; 

} 

fos.write(buf, 0, numread); 
}while (true); 


7) 打开 铃声 管理 对 象 RingtonManager 进行 铃声 选择 ， 通 过 Intent 对 象 intent 来 设置 铃 


声 ， 然 后 设置 显示 铃声 的 文件 夹 和 显示 铃声 开头 。 如 有 果 有 异常 则 输出 异常 。 具 体 代码 如 下 。 
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上 打开 RingtonManager 进行 铃声 选择 */ 
String uri = null; 
if(bIfExistRingtoneFolder(strRingtoneFolder)) 


{ 


请 设 置 铃声 


Intent intent = new Intent( RingtoneManager. 
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ACTION RINGTONE PICKERJ); 

启 设 置 显示 铃声 的 文件 来 */ 

intent.putExtra( Ringtone Manager.EXTRA RINGTONE TYPE, 
RingtoneManager.TYPE RINGTONE); 

族 设 置 显示 铃声 开头 */ 

intent.putExtra( Ringtone Manager.EXTRA RINGTONE TITLE, 
"设置 铃声 

if( uri != null) 

{ 

intent.putExtra( RingtoneManager. 
EXTRA RINGTONE EXISTING URI, Uri.parse( uri)); 


} 


else 


{ 


intent.putExtra( RingtoneManager. 
EXTRA RINGTONE EXISTING URI, (Uri)null); 


} 
startActivityForResult(intent, RINGTONE PICKED); 


try 
{ 


is.close(); 


} 


catch (Exception ex) 


{ 
Log.e(APP TAG, "error: " + ex.getMessage(), ex); 


} 


8) 定义 方法 onActivityResult()， 能 够 根据 用 户 选 择 的 铃声 设置 保存 对 应 的 信息 。 当 选择 
完毕 后 ， 会 再 次 返回 选择 Activity。 有 具体 代码 如 下 。 


mm 


(QOverride 
protected void onActivityResult(int requestCode, 
int resultCode, Intent data) 


{ 
if (resultCode != RESULT OK) 
{ 
return; 
} 
switch (requestCode) 
{ 
case (RINGTONE PICKED): 
try 
{ 
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Uri pickedUri = data.getParcelableExtra 
(RingtoneManager.EXTRA RINGTONE PICKED UR); 
if(pickedUri!=null) 
{ 
RingtoneManager.setActualDefaultRingtoneUri 
(example7.this,RingtoneManager.TYPE RINGTONE, 


pickedUri); 
} 
} 
catch(Exception e) 
1 
e.printStack Trace(); 
} 
break; 
default: 
break; 
} 
super.onActivityResult(requestCode, resultCode, data); 


} 


Wea 


9) 定义 bIfExistRingtoneFolder0) 方 法 ， 用 于 判断 是 否 包 含 了 “/sdcard/music/ringtones” 文 
件 夹 。 具 体 代 码 如 下 。 


访 判 断 是 否 包 含 /sdcard/music/ringtones 文件 夹 */ 
private boolean bIfExistRingtoneFolder(String strFolder) 


{ 


boolean bReturn = false; 


File f= new File(strFolder); 
if(!f.exists()) 
{ 
/* 创 建 /sdcard/music/ringtones 文件 夹 */ 
if(f.mkdirs()) 
{ 
bReturn = true; 
} 
else 


{ 
bReturn = false; 


} 


else 


{ 


bReturn = true; 


} 


return bReturn; 
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} 
执行 后 会 先 显示 一 个 下 载 界面 ， 如 图 8-10 所 示 。 单 击 “ 下 载 音 乐 ”按钮 后 开始 下 载 指定 
的 MP3 文件 ， 下 载 完成 后 会 弹出 “铃声 设置 ”界面 ， 如 图 8-11 所 示 。 选 择 一 种 选项 ， 并 单 
击 “OK” 按钮 后 ， 完 成 铃声 设置 。 


@ 设置 铃声 


© 


Silent 


Default ringtone 


[=] 转 古 贱 恒 1:26Aw 


practice7 


Beat Plucker 


http://www.lrn.cn/zywh/xyyY/ 
yyxs/200805/ Bell Phone 
W020080505536315331317.mp3 


下 载 音乐 … Bentley Dubs 


| OK | Cancel 


图 8-10 初始 界面 图 8-11 铃声 设置 界面 


© ©© © 
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以 从 网 络 中 直接 下 载 一 个 图 片 文 件 ， 并 设置 为 手机 屏幕 背景 。 在 本 节 的 内 容 中 ， 将 通 
过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 远程 下 载 图 片 作为 屏幕 背景 的 基本 流程 。 本 实例 
的 源 代码 保存 在 “光盘 \daima\8\example8”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


8.8.1 实现 原理 


在 本 实例 中 ， 可 以 远程 获取 网 络 中 的 一 幅 图 片 ， 并 将 这 幅 图 片 作 为 手机 屏幕 的 背景 。 在 
具体 实现 上 ,将 通过 InputStream 传 到 ContextWrapper 中 重 写 的 setwallpaper0 方 法 ， 其 中 传 入 
的 参数 是 URLConnection.getInputStream0 的 Stream 内 容 。 


口 


A 


8.8.2 ”具体 实现 


本 实例 的 主 程序 文件 是 example8.java， 下 面 开 始 介绍 其 实现 流程 。 
1) 先 加 载 相 关 类 ， 然 后 声明 系统 中 需要 的 私有 变量 ， 有 具体 代码 如 下 。 


public class example8 extends Activity 


{ 
Wt 局 量 声明 */ 


private Button mButton1; 


private Button mButton2; 
private EditText mEditText; 
private ImageView mImageView; 
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private Bitmap bm; 


2) 通过 findViewByld 初始 化 各 个 对 象 ， 单 击 按钮 1， 通 过 


代码 如 下 。 


(@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


庆 初始 化 对 象 迪 

mButtonl =(Button) fmndViewById(R.id.myButtonl ); 
mButton2 =(Button) fmndViewById(R.id.myButton2); 
mEditText = (EditText) fndViewById(R.id.myEdit); 
mlmageView = (ImageView) findViewById(R.id.myImage); 
mButton2.setEnabled(false); 


谨 预览 图 片 的 按钮 * 
mButtonl.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
String path=mEditText.getText().toString(); 
if(path.equals("")) 
{ 
showDialog(" 网 址 不 可 为 空白 1"); 
} 
else 
{ 
庶 传 入 type=1 表示 预览 图 片 六 
setImage(path, 1); 
} 
} 
)D); 


3) 单 击 按钮 2， 通 过 mButton1.setOnClickListener 将 图 片 设 为 桌面 。 


白 提 示 ， 否 则 传 入 “type=2” 表 示 设 置 桌面 。 具 体 代码 如 下 。 


翌 将 图 片 设 为 桌面 的 Button */ 
mButton2.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 


406 四 看 


二 mButton1.setOnClickListener 
来 实现 图 片 预览 。 网 址 为 空 则 输出 空白 提示 ， 不 为 空 则 传 入 “type=1” 表 示 预 览 图 片 。 具 体 


人 


网 址 为 空 则 输出 空 


泌 
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try 


一 一 


String path=mEditText.getText().toString(); 
if(path.equals("")) 
{ 
showDialog(" 网 址 不 可 为 空白 !"); 
} 
else 
{ 
访 传 入 type=2 表示 设置 桌面 */ 
setImage(path,2); 


} 


catch (Exception e) 


showDialog(" 读 取 错 误 ! 网 址 可 能 不 是 图 片 或 网 址 错误 !"); 


bm = null; 


mlmageView.setImageBitmap(bm); 
mButton2.setEnabled(false); 
e.printStack Trace(); 


上 
} 


4) 定义 setImage(String path,int type) 方 法 ， 用 于 预览 图 片 并 设置 为 代 面 。 首 
然后 设置 为 介面。 有 异常 则 输出 对 应 提示 。 具 体 代码 如 下 。 


入 预览 图 片 并 设置 为 桌面 的 方法 */ 
private void setImage(String path,int type) 
{ 
try 
{ 
URL url = new URL(path); 


URLConnection conn = url.openConnection(); 


页 
让 


预览 氏 


~ 


conn.connect(); 
if(type==1) 
{ 
庆 预览 图 片 % 
bm = BitmapFactory.decodeStream(conn.getInputStream()); 


mlmageView.setImageBitmap(bm); 
mButton2.setEnabled(true); 


} 
else if(type==2) 
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{ 
/* 设置 为 桌面 */ 


example8.this.setWallpaper(conn.getImnputStream()); 


bm = null; 
mlmageView.setImageBitmap(bm); 
mButton2.setEnabled(false); 
showDialog(" 桌 面 背 景 设置 完成 !"); 


} 

) 

catch (Exception e) 

{ 
showDialog(" 读 取 错 误 ! 网 址 可 能 不 是 图 片 或 网 址 错误 !"); 
bm = null; 


mlmageView.setImageBitmap(bm); 
mButton2.setEnabled(false); 
e.printStack Trace(); 


} 


5) 定义 showDialog(String mess) 方 法 ， 用 于 弹出 对 话 框 ， 单 击 后 完成 背景 设置 。 具 体 代 
码 如 下 。 


诺 弹出 对 话 框 的 方法 *%/ 

private void showDialog(String mess){ 
new AlertDialog.Builder(example8.this).setTitle("Message") 
.SetMessage(mess) 


人 1 


.SetNegativeButton(" 确 定 


{ 
public void onClick(DialogInterface dialog, int which) 


, new DialogInterface.OnClickListener() 


最 后 还 需要 在 文件 AndroidManifest.xml 中 设置 SET_WALLPAPER 权限 和 INTERNET 权 
限 ， 且 体 代码 如 下 。 


<uses-permission android:name="android.permission.SET WALLPAPER"/> 
<uses-permission android:name="android.permission.INTERNET"/> 


执行 后 将 显示 一 个 输入 框 和 2 个 按钮 ， 如 图 8-12 所 示 。 输 入 图 片 网 址 并 单 击 “预览 ” 
按钮 后 ， 可 以 查看 此 图 片 ， 如 图 8-13 所 示 。 单 击 “ 设 置 ” 按 钮 后 ， 将 设置 此 图 片 为 屏幕 
背景 
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[= 国 驯 贱 介 1:36Aw [= 园区 贱 多 1:33 


practice8 


practice8 
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图 8-12 初始 效果 图 8-13 ”设置 为 背景 
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文件 上 传 对 于 广大 读者 来 说 并 不 陌生 ， 在 手机 中 同样 可 以 实现 文件 上 传 功能 。 在 本 节 的 
内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 实现 文件 上 传 的 基本 流程 。 本 实例 
的 源 代码 保存 在 “光盘 :daimaNg\example9”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


8.9.1 实现 原理 
在 Web 应 用 中 ， 通 常 通过 一 个 表单 和 上 传 程序 来 实现 文件 上 传 。 查 看 下 面 的 代码 。 


<FORM METHOD="POST" ACTION="do upload.jsp" ENCTYPE="multipart/form-data"> 
<input type="FILE" name="FILE1" size="30"> 

<input type="submit" name="Submit" value=" 上传 它 ! "> 

</FORM> 


上 上 述 代 码 是 一 个 上 传 表 单 ， 供 用 户 选择 上 传 文件 ， 并 通过 文件 do_upload.jsp 实现 上 传 处 
理 。 在 本 实例 中 ， 将 模拟 上 述 处 理 过 程 ， 以 Post 方式 对 服务 器 上 的 接收 程序 发 出 请 求 ， 触 发 
该 程序 运行 文件 写 入 服务 器 的 动作 。 在 实现 本 实例 前 ， 需 要 搭建 Java 的 服务 器 ， 并 在 服务 器 
预先 编写 一 个 接收 文件 程序 ， 并 在 手机 目录 中 准备 要 上 传 的 资料 。 在 此 ， 设 置 要 上 传 的 文件 
路 径 如 下 。 


data/data/irdc.example9/image.jpg 


然后 准备 上 传 处 理 文件 upload.jsp， 其 代码 将 不 介绍 ， 在 网 中 比比 皆 是 


8.9.2 具体 实现 
本 实例 的 主 程序 文件 是 example9.java， 下 面 开始 介绍 其 实现 流程 。 
1) 先 加 载 相关 类 ， 然 后 分 别 声明 变量 newName、uploadFile 和 actionUrl， 有 具体 代码 如 下 
所 示 。 


public class example9 extends Activity 


{ 
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从 变量 声明 
* newName: 上 传 后 在 服务 器 上 的 文件 名 称 
* uploadFile: 要 上 传 的 文件 路 径 
* actionUrl: 服务 器 上 对 应 的 程序 路 径 */ 


fh 


private String newName="image.jpg"; 


private String uploadFile="/data/data/irdc.example9/image.jpg"; 
private String actionUrl="http://127.127.0.1/upload/upload.jsp"; 
private TextView mTextl; 

private TextView mText2; 

private Button mButton; 


2) 通过 mTextl 获取 文件 路 径 ， 根 据 mText2 设置 上 传 网 址 ， 单 击 按钮 后 调用 上 传 方法 
uploadFile()。 具 体 代码 如 下 所 示 。 


(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


mTextl = (TextView) fndViewById(R.id.myText2); 
mTextl.setText(" 文 件 路 径 : \n"+uploadFile); 
mText2 = (TextView) fmndViewById(R.id.myText3); 
mText2.setText(" 上传 网 址 : "+actionUrD; 
/* 设置 mButton 的 onClick 事件 处 理 */ 
mButton = (Button) findViewById(R.id.myButton); 
mButton.setOnClickListener(new View.OnClickListener() 
{ 

public void onClick(View v) 

{ 

uploadFile(); 

} 

D); 
} 


3) 定义 方法 uploadFile0， 用 于 将 文件 上 传 至 远 端 服务 器 。 流 程 如 下 。 


口 第 一 步 : 设置 传送 的 method=POST。 
口 第 二 步 : 设置 设置 DataOutputStream。 
口 第 三 步 : 获取 文件 的 FileInputStream。 
口 第 四 步 : 设置 每 次 写 入 1024bytes。 

口 第 五 步 ， 从 文件 中 读 取 数据 至 缓冲 区 。 
口 第 六 步 : 获取 Response 的 内 容 。 

口 第 七 步 : 把 Response 在 Dialog 中 显示 。 


L 体 代码 如 下 。 


/* 上 传 文件 至 Server 的 方法 六 
private void uploadFile() 
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String end = "\rn"; 
String twoHyphens = "--"; 
String boundary = "*****"; 
try 
{ 
URL url =new URL(actionUr)); 


HttpURLConnection con=(HttpURLConnection)url.openConnection(); 


/# 允许 Input、Output， 不 使 用 Cache */ 
con.setDoInput(true); 


con.setDoOutput(true); 
con.setUseCaches(false); 

人 # 设置 传送 的 method=POST */ 
con.setRequestMethod("POST"); 
/* setRequestProperty */ 


con.setRequestProperty("Connection", "Keep-Alive"); 

con.setRequestProperty("Charset", "UTF-8"); 

con.setRequestProperty("Content-Type", 
"multipart/form-data;boundary="+boundary); 

/# 设置 DataOutputStream */ 

DataOutputStream ds = 


new DataOutputStream(con.getOutputStream()); 
ds.writeBytes(twoHyphens + boundary + end); 
ds.writeBytes("Content-Disposition: form-data; "十 
"name=\"filel\";filename=\""+ 
newName +"\"" + end); 
ds.writeBytes(end); 


/* 取得 文件 的 FileInputStream */ 

FileInputStream fStream = new FileInputStream(uploadFile); 
/* 设置 每 次 写 入 1024bytes */ 

int bufferSize = 1024; 

byte[] buffer = new byte[bufferSize]; 


int length = -1; 
谨 从 文件 读 取 数据 至 缓冲 区 */ 
while((length = fStream.read(buffer)) != -1) 
{ 
/* 将 资料 写 入 DataOutputStream 中 对 
ds.write(buffer, 0, length); 


} 
ds.writeBytes(end); 
ds.writeBytes(twoHyphens + boundary + twoHyphens + end); 


/* close streams */ 
fStream.close(); 
ds.flush(); 


/* 取得 Response 内 容 */ 
InputStream is = con.getInputStream(); 
int ch; 
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StringBuffer b =new StringBuffer(); 
while( (ch=is.read() ) (= -1 ) 
{ 
b.append( (char)ch ); 

} 
/* 将 Response 显示 于 Dialog */ 
showDialog(b.toString().trim()); 
/* 关闭 DataOutputStream */ 
ds.close(); 

} 

catch(Exception e) 

{ 
ShowDialog(""+e); 


} 


} 


4) 定义 方法 showDialog(String mess)， 用 于 显示 提示 对 话机 


it 
o 


具体 代码 如 下 。 


/* 显示 Dialog 的 method */ 
private void showDialog(String mess) 
{ 
new AlertDialog.Builder(example9.this).setTitle("Message") 

.set Message(mess) 

.SetNegativeButton(" 确 定 ",new DialogInterface.OnClickListener() 

{ 

public void onClick(DialogInterface dialog, int which) 


执行 后 的 效果 如 图 8-14 所 示 。 单 击 “ 开 始 上 传 ” 按钮 后 ， 能 够 将 指定 文件 上 传 到 服务 器 。 


动 天 天 1:47AM 


EXarmpJe9 


文件 上 传 系统 


文件 路 径 : 
/data/data/irdc.shili9/image.jpg 


上 传 网 址 : 
httpV7127.127.0.17upload/uploadjsp 


开始 上 传 
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图 8-14 执行 效果 
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RSS 是 一 个 开放 的 新 闻 模 式 ，RSS 是 在 线 共 享 内 容 的 一 种 简易 方式 ， 也 叫 聚 合 内 容 
(Really Simple Syndication，RSS)。 通 常 在 时 效 性 比较 强 的 内 容 上 使 用 RSS 订阅 能 更 快速 获 
取信 息 ， 网 站 提供 RSS 输出 ， 有 利于 用 户 获取 网 站 内 容 的 最 新 更 新 。 在 本 节 的 内 容 中 ， 将 通 
过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 查看 指定 RSS 新 闻 的 基本 流程 。 本 实例 的 源 代码 
保存 在 “光盘 :\daima\8\example10”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


8.10.1 实现 原理 

在 使 用 RSS 订阅 时 ， 通 常 通过 网 站 提供 的 “订阅 RSS” 连 接 或 小 图 标 实现 ， 当 单 击 连接 
后 ,会 弹出 包含 RSS 内容 的 页 面 ， 此 页 面 的 网 址 是 网 站 的 RSS 网 址 。 当 连接 后 ， 服 务 器 端 会 
返回 RSS 标准 规格 的 XML 文件 ， 只 要 按照 统一 格式 来 解析 XML 文件 ， 就 可 以 得 到 RSS 内 
的 相关 信息 。 
在 本 实例 中 ， 用户 只 需要 输入 一 个 RSS Feed 网 址 ， 通 过 SAXParser 解析 后 就 可 以 直接 在 
手机 上 浏览 在 线 新 闻 。 


8.10.2 具体 实现 


本 实例 的 主 程序 文件 包括 example10.java、example10_1.java、example10 2.java、News.java、 
MyHandlerjava 和 MyAdapterjava， 下 面 开 始 分 别 介 绍 其 实现 流程 。 

1. 主 程序 example10.java 

主 程序 example10.java 中 以 EditText 来 作为 输入 RSS 连接 组 件 。 当 输入 网 址 后 , 单 击 “ 解 
析 ” 按 钮 后 ， 按 钮 的 onClick ( 单 击 事件 ) 会 被 触发 ， 运 行 EditText 的 空白 检查 。 当 检查 无 误 
后 ， 将 输入 的 网 址 写 入 Bundle《〈 绑 定 ) 对 象 中 ， 再 将 Bundle 对 象 指派 给 Intent， 并 通过 
startActivityForResult() 方 法 来 触发 example10_1 这 个 Activity。 

主 程序 example10.java 的 实现 代码 如 下 。 


public class example10 extends Activity 
{ 

/* 忆 量 声 明 */ 

private Button mButton; 

private EditText mEditText; 


(@Override 

public void onCreate(Bundle savedInstanceState) 

{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
雍 初始 化 对 象 沁 
mbEditText=(EditText) fndViewById(R.id.myEditb); 
mButton=(Button) fndViewById(R.id.myButton); 
旋 设置 按钮 的 单 击 事件 */ 
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mButton.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
public void onClick(View v) 
{ 
String path=mEditText.getText().toString(); 
if(path.equals("")) 
{ 
showDialog(" 网 址 不 可 为 空白 1"); 
} 
else 
{ 
/* 新 建 一 个 Intent 对 象 ， 并 指定 类 */ 
Intent intent = new Intent(); 
intent.setClass(examplel10.this,example10 1.class); 
放 new 一 个 Bundle 对 象 ， 并 将 要 传递 的 数据 传 入 * 
Bundle bundle = new Bundle(); 
bundle.putString("path",path); 
/* 将 Bundle 对 象 assign 给 Intent */ 
intent.putExtras(bundle); 
庆 调用 Activity EX08 13 1*/ 
startActivityForResult(intent,0); 
} 
} 
)); 
} 
/* 履 盖 onActivityResult(O*/ 
(@Override 
protected void onActivityResult(int requestCode,int resultCode, Intent data) 
{ 
switch (resultCode) 
{ 
case 99: 


雍 返回 错误 时 以 Dialog 显示 */ 
Bundle bunde = data.getExtras(); 
String error = bunde.getString("error"); 
showDialog(error); 
break; 

default: 
break; 


从 显示 对 话 框 的 方法 光 
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private void showDialog(String mess){ 
new AlertDialog.Builder(example10.this).setTitle("Message") 
.setMessage(mess) 
.SetNegativeButton(" 人 确定 ", new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog, int which) 


2. 文件 example10_1.java 

文件 example10_1.java 是 一 个 ListActivity, 是 通过 主 程序 example10.java 来 调用 的 ,用 于 
显示 订阅 的 RSS 内 容 列表 。 其 实现 流程 如 下 。 

1) 加 载 相 关 类 ， 分 别 声 明 变 量 mText、title 和 li。 有 具体 代码 如 下 所 示 。 


public class example10 1 extends ListActivity 
{ 

/*# 变量 声明 *%/ 

private TextView mText; 


private String title=""; 
private List<News> li=new ArrayList<News>(); 


2) 设置 layout 为 newslist.xml， 取 得 Intent 中 的 Bundle 对 象 ， 并 取得 Bundle 对 象 中 的 数 
据 ， 然 后 调用 getRssO 取 得 解析 后 的 List。 具 体 代 码 如 下 。 
(QOverride 


public void onCreate(Bundle savedInstanceState) { 


super.onCreate(savedInstanceState); 


/# 设置 layout 为 newslist.xml */ 


setContentView(R.layout.newslist); 


mText=(TextView) fndViewById(R.id.myText); 
/* 取得 Intent 中 的 Bundle 对 象 */ 
Intent intent=this. getIntent(); 

Bundle bunde = intent.getExtras(); 

族 取得 Bundle 对 象 中 的 数据 */ 
String path = bunde.getString("path"); 
/* 调用 getRss0 取 得 解析 后 的 List */ 
li=getRss(path); 

mText.setText(title); 

上 # 设置 自 定义 的 MyAdapter */ 
SetListAdapter(new MyAdapter(this,li)); 
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3) 定义 onListttemClick() 方 法 ， 作 为 监听 ListItem 被 单 击 时 要 做 的 动作 。 流 程 如 下 。 

口 第 一 步 : 获取 News 对 象 。 

口 第 二 步 : 新 建 一 个 Intent 对 象 ， 并 指定 其 类 名 。 

口 第 三 步 : 新 建 一 个 Bundle 对 象 ， 并 将 要 传递 的 数据 传 入 。 

口 

口 


第 四 步 : 将 Bundle 对 象 分 配给 Intent。 
第 五 步 : 调用 Activity example10 2。 
只 体 代 码 如 下 。 


谨 设置 ListItem 被 点 击 时 要 做 的 动作 */ 
@Override 
protected void onListItemClick(ListView 1,View v,int position, long id) 
{ 
/* 取得 News 对 象 */ 
News ns=(News)li.get(position); 
/* new 一 个 Intent 对 象 ， 并 指定 class */ 


Intent intent = new Intent(); 


intent.setClass(example10 1.this,example10 2.class); 

旋 new 一 个 Bundle 对 象 ， 并 将 要 传递 的 数据 传 入 六 
Bundle bundle = new Bundle(); 
bundle.putString("title",ns.getTitle()); 
bundle.putString("desc",ns.getDesc()); 
bundle.putString("link",ns.getLink()); 

/* 将 Bundle 对 象 分 配给 Intent */ 
intent.putExtras(bundle); 

/* 调用 Activity example10 2 */ 

startActivity(intent); 


} 


4) 定义 getRss(String patb) 方 法 ， 用 于 解析 XML。 流 程 如 下 。 
口 第 一 步 : 通过 url 获取 地 址 。 


生生 一 上 


第 二 步 : 创建 SAXParser 对 象 和 XMLReader 对 象 。 
第 三 步 : 设置 自 定 义 的 MyHandler 给 XMLReader。 
第 四 步 : 解析 XML。 

第 五 步 : 取得 RSS 标题 与 内 容 列 表 。 
人体 代码 如 下 。 


和 # 解析 XML 的 方法 尺 

private List<News> getRss(String path) 

{ 
List<News> data=new ArrayList<News>(); 
URL url = null; 
try 


{ 
url= new URL(path); 


口 
口 
口 
口 


416 大 下 


第 8 章 网 络 应 用 


/* 产生 SAXParser 对 象 */ 

SAXParserFactory spf = SAXParserFactory.newInstance(); 
SAXParser sp = spf.newSAXParser(); 

/六 产生 XMLReader 对 象 */ 

XMLReader xr = sp.getXMLReader(); 


旋 设置 自 定义 的 MyHandler 给 XMLReader */ 
MyHandler myExampleHandler = new MyHandler(); 


xr.setContentHandler(myExampleHandler); 
庆 解析 XML */ 

xr.parse(new InputSource(url.openStream())); 
入 取得 RSS 标题 与 内 容 列表 */ 

data =myExampleHandler.getParsedData(); 


title=myExampleHandler.getRssTitle(); 


} 


5) 有 异常 则 输 


错误 提示 对 话 和 


TH 


， 有 具体 代码 如 下 。 


catch (Exception e) 


上 # 发 生 错误 时 返回 结果 回 上 一 个 activity */ 
Intent intent=new Intent(); 

Bundle bundle = new Bundle(); 
bundle.putString("error",""+e); 


intent.putExtras(bundle); 

上 # 错误 的 返回 值 设 置 为 99 */ 
examplel10 1.this.setResult(99, intent); 
example10 1.this.finish(); 


} 


return data; 


} 
} 


3. 文件 example10_2.java 


文件 example10 2.java 由 example10_1 唤起 ,上 


] 于 显示 上 一 个 Activity 所 单 击 的 新 闻 内 容 。 


当 程序 被 唤起 后 ， 会 首先 从 Bundle 对 象 中 获取 信息 的 标题 、 链 接 和 描述 ， 并 显示 在 画面 中 。 


并 以 Linkify.addLinks() 方 法 将 link 设置 为 一 个 WEB _URLS 形式 的 链接 。 当 用 户 单 击 链接 后 
接 打 开 Web 浏览 器 来 浏览 网 页 。 其 具体 实现 代码 如 下 。 


会 通过 设置 的 网 址 直 


package irdc.example10; 


六 
import irdc.example10.R; 


import android.app.Activity; 


import android.content.Intent; 


import android.os.Bundle; 
import android.text.util.Linkify; 
import android.widget.TextView; 
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public class example10 2 extends Activity 


{ 


} 


入 变量 声明 导 

private TextView mTitle; 
private TextView mDesc; 
private TextView mLink; 


(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
* 设置 


layout 为 newscontent.xml */ 


setContentView(R.layout.newscontent); 


/* 初始 化 对 象 */ 


mTitle=(TextView) findViewByld(R.id.myTitle); 
mDesc=(TextView) fndViewById(R.id.myDesc); 
mLink=(TextView) findViewById(R.id.myLink); 


/* 取得 Intent 中 的 Bundle 对 象 */ 

Intent intent=this. getIntent(); 

Bundle bunde = intent.getExtras(); 

族 取得 Bundle 对 象 中 的 数据 */ 
mTitle.setText(bunde.getString("title")); 
mDesc.setText(bunde.getString("desc")+"...."); 
mLink.setText(bunde.getString("link")); 

旋 设置 mLink 为 网 页 连接 */ 
Linkify.addLinks(mLink,Linkify. WEB_URLS) 


4. 文件 News.java 


文件 News.java 是 一 个 JavaBean 类 ， 用 于 存放 每 一 篇 新 闻 信 息 。 每 一 个 News 对 银 代 于 
萌 述 、 网 站 链接 和 发 布 时 间 这 4 个 属性 。 
类 中 的 方法 都 是 以 setAAAO 和 getAAA0O 方 式 来 命名 的 ， 所 以 在 Newsijava 中 用 
setAAA(0O 来 设置 属性 值 ， 或 通过 getAAA0 来 获取 属 


了 一 条 新 


JavaBean 
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》 


La 


恒 ， 在 News 对 象 中 定义 了 新 闻 的 标题 、 


package irdc.example10; 


public class News 


{ 


private String _title=""; 
private String _link=""; 
private String _desc=""; 
private String _date=""; 


public String getTitle() 
{ 


必 


E 值 。 具 体 代码 如 下 。 


| 
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return title; 
} 
public String getLink() 
{ 
return _link; 
} 
public String getDesc() 
{ 
return desc; 
} 
public String getDate() 
{ 
Tetum _date; 
} 
public void setTitle(String title) 
{ 
_title=title; 
} 
public void setLink(String link) 
{ 
_link=link; 
} 
public void setDesc(String desc) 
{ 
_desc=desc; 
} 
public void setDate(String date) 
{ 
_date=date; 
} 
} 


5. 文件 MyAdapter.java 
在 文件 MyAdapterjava 中 定义 了 Adapter 对 象 ， 它 继承 自 android.widget.BaseAdapter， 用 


于 设置 ListView 中 要 显示 的 信息 ， 用 文件 news_row.xml 作为 布局 。 具 体 代码 如 下 。 


/* 自 定义 的 Adapter， 继 承 android.widget.BaseAdapter */ 
public class MyAdapter extends BaseAdapter 
{ 

局 变量 声明 沁 

private LayoutInflater mInflater; 


private List<News> items; 


/* MyAdapter 的 构造 器 ， 传 递 两 个 参数 六 
public MyAdapter(Context context,List<News> it) 
{ 
庆 参数 初始 化 * 
mInfater=LayoutImflater.ffrom(context); 
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items = it; 
} 
/* 因 继 承 BaseAdapter， 需 重 写 以 下 方法 */ 
@Override 
public int getCount() 


{ 


return items.size(); 


} 


(@Override 
public Object getItem(int position) 


{ 


return items.get(position); 


} 


(@Override 
public long getItemld(int position) 
{ 


return position; 


} 


(@Override 
public View getView(int position,View convertView,ViewGroup par) 


{ 
ViewHolder holder; 


if(convertView == null) 
{ 
/* 使 用 自 定义 的 news_row 作为 Layout */ 
convertView = mInflater.inflate(R.layout.news_row, null); 
/* 初始 化 holder 的 text 与 icon */ 
holder = new ViewHolder(); 
holder.text = (TextView) convertView.findViewBylId(R.id.text); 
convertView.setTag(holder); 
} 
else 
{ 
holder = (ViewHolder) convertView.getTag(); 
} 
News tmpN=(News)items.get(position); 
holder.text.setText(tmpN.getTitle()); 


return convertView; 


} 


/* class ViewHolder */ 
private class ViewHolder 


{ 


TextView text; 
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6. 文件 MyHandler.java 
在 文件 MyHandlerjava 中 定义 了 MyHandler 对 象 ， 它 继承 于 org.xml.sax.helpers. 


下 面 开 始 讲解 文件 MyHandlerjava 的 具体 实现 流程 。 
1) 加 载 相 关 类 ， 然 后 分 别 声明 各 个 变量 。 具 体 代码 如 下 。 


Et 


public class MyHandler extends DefaultHandler 


{ 


2) 返 


。 然 后 调用 startDocument() 方 法 开始 解析 操作 。 当 解析 结束 时 ， 调 
解析 到 Element 开头 时 ， 调 用 startElement0) 方 法 。 有 具体 代码 如 下 。 


变量 声明 */ 


private boolean in item = false; 


private boolean in title = false; 

private boolean in link = false; 

private boolean in_desc = false; 

private boolean in date = false; 

private boolean in mainTitle = false; 

private List<News> li; 

private News news; 

private String title=""; 

private StringBuffer buf~new StringBuffer(); 


回 将 转换 成 List<News> 的 XML 数据 ， 通 过 getRssTitle(0) 方 法 返回 解析 
] endDocument() 方 法 。 


/* 将 转换 成 List<News> 的 XML 数据 返回 */ 
public List<News> getParsedData() 
{ 
return li; 
} 
族 返回 解析 出 的 RSS 标题 */ 
public String getRssTitle() 
{ 
return title; 
} 
放 义 ML 文件 开始 解析 时 调用 此 方法 */ 
(QOverride 


public void startDocument() throws SAXException 


{ 
li=new ArrayList<News>(); 
} 
放 XML 文件 结束 解析 时 调用 此 方法 * 
(QOverride 
public void endDocument() throws SAXException 
{ 
} 


/* 解析 到 Element 的 开头 时 调用 此 方法 * 


bh 的 RSS 标 
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(@Override 


public void startElement(String namespaceURI, String localName, 
String qName, Attributes atts) throws SAXException 


{ 


if (localName.equals("item")) 


{ 
this.in item = true; 
/* 解析 到 item 的 开头 时 new 一 个 News 对 象 */ 
news=new News(); 


} 


else if (localName.equals( "title")) 


{ 
iftthis.in_item) 
{ 
this.in title = true; 
} 


else 


{ 


this.in mainTitle = true; 


} 


} 
else if (localName.equals("link")) 


iftthis.in item) 
{ 
this.in link = true; 
} 
} 


else if (localName.equals("description")) 


{ 
iftthis.in_item) 
{ 
this.in_desc = true; 
} 


} 
else if (localName.equals("pubDate")) 


{ 
iftthis.in item) 
{ 
this.in date = true; 
} 
} 
} 


3 ) 当 解 析 到 Element〔 元 素 ) 的 结尾 时 调用 endElement0 方 法 ， 流 程 如 下 。 

口 第 一 步 : 将 解析 到 Item 的 结尾 时 将 News 对 象 写 入 List 中 。 

口 第 二 步 : 根据 Item 选项 分 别 设置 News 对 象 的 标题 ， 设 置 RSS 的 标题 ， 设 置 News 
对 象 的 链接 ， 设 置 News 对 象 的 描述 ， 设 置 News 对 象 的 pubDate 时 间 。 

人 体 代 码 如 下 。 
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翌 解析 到 Element 的 结尾 时 调用 此 方法 */ 
(@Override 
public void endElement(String namespaceURI, String localName, 


String qName) throws SAXException 
{ 
if (localName.equals("item")) 
{ 
this.in item = false; 
上/# 解析 到 item 的 结尾 时 将 News 对 象 写 入 List 中 */ 
li.add(news); 
} 
else if (localName.equals( "title")) 
{ 
iftthis.in_item) 
{ 
/* 设置 News 对 象 的 标题 */ 
news.setTitle(buf.toString().trim()); 
buf.setLength(0); 
this.in title = false; 
} 
else 
{ 
/* 设置 RSS 的 标题 */ 
title=buf.toString().trim(); 
buf.setLength(0); 
this.in mainTitle = false; 
} 
} 
else if (localName.equals("link")) 
{ 
iftthis.in_item) 


{ 


/* 设置 News 对 象 的 链接 */ 
news.setLink(buf.toString().trim()); 
buf.setLength(0); 
this.in link = false; 
} 
9 
else if (localName.equals("description")) 
{ 
这 in item) 
{ 
/* 设置 News 对 象 的 描述 */ 
news.setDesc(buf.toString().trim()); 
buf.setLength(0); 
this.in desc = false; 


} 
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} 
else if (localName.equals("pubDate")) 


{ 
if(in item) 
{ 
/* 设置 News 对 象 的 时 间 */ 
news.setDate(butf.toString().trim()); 
bufsetLength(0); 
this.in date = false; 
} 
} 
} 


4) 定义 方法 characters()， 用 于 获取 Element 开头 和 结尾 中 间 的 字符 串 。 具 体 代 码 如 下 。 


举 取得 Element 的 开头 结尾 中 间 夹 的 字符 串 */ 
(QOverride 
public void characters(char ch[], int start, int length) 
{ 
if(this.in_ itemllthis.in mainTitle) 
{ 
/* 将 char[] 添 加 StringBuffer */ 
buf.append(ch,start,length); 
} 
} 
》 


执行 后 的 效果 如 图 8-15 所 示 。 在 文本 框 中 输入 RSS 网 址 http://rss.sina.com.cn/news 
/marquee/ ddt.xml， 然 后 单 击 “开始 解析 ”按钮 后 ,会 在 屏幕 中 列表 显示 RSS 新 闻 ， 如 图 8-16 
所 示 。 单 击 某 条 新 闻 后 ， 会 显示 此 新 闻 的 详情 ， 如 图 8-17 所 示 。 


全 555 


4: 全 


[= 园 古 贱 多 2:14 


example10 


新 闻 中 心 

体育 ] 网 友 串 中 世界 杯 国足 比分 中 万 元 另 1 人 千 倍 
美国 1-1(06/27 09:52) 

》 [体育 ] 篮 网 看 好 探花 阿 联 被 摆 上 货架 太空 易 急需 
证 明 自 己 (06/27 09:30) 

P [科技 JjMySpace 失 血 不 断 无 所 适 从 : 难 逃 中 国 属 
加 (06/27 09:04) 


example10 


设置 需要 的 RSS 连 接 : 


开始 解析 


科技 ] 炮 总 “无 氰 空调 ” : 董 明珠 的 难 言 之 

隐 (06/27 09:00) 

P [科技] 网 络 游戏 霸王 条 款 时 代 结 束 : 互联 网 国家 
行动 (06/27 08:58) 

国际 ] 加 拿 大 数 千 人 参加 G20 示 威 与 警察 冲 

突 (图 06/27 08:38) 

P [国内] 云南 马龙 通 遇 特大 洪灾 受 困 灾 民 积极 自 
救 (组 图 06/27 08:07) 

” [国内] 云南 马龙 水 库 漫 坝 县 城 被 淹 4000 余 人 被 
困 ( 图 06/27 07:55) 

国际 ] 欧 洲 阿 丽 亚 娜 火箭 成 功 发 射 两 颗 卫 

星 (06/27 06:35) 

P [社会 ] 小 伙 冒 充 单身 接近 开 名 车 女子 骗 

得 200 万 (06/27 03:55) 


社会 ] 北 京 第 二 代 胶 这 公 寓 被 房 主 拆除 将 建 第 三 
De 和 


图 0 


pal 
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图 8-15 ”初始 效果 
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在 使 用 智能 手机 时 ， 经 常 需要 直接 利用 手机 来 下 载 网 络 中 的 软件 程序 ， 然 后 安装 这 个 下 
载 的 软件 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 过 程 ， 介 绍 在 Android 中 远程 下 载 
Android 软件 的 基本 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\8\examplel1”， 下 面 开始 讲解 
本 实例 的 具体 实现 流程 。 


8.11.1 APK 简 介 


APK 是 Android Package 的 缩写 ， 即 Android 安装 包 。APK 是 类 似 Symbian Sis 或 Sisx 的 
文件 格式 。 通 过 将 APK 文件 直接 传 到 Android 模拟 器 或 Android 手机 中 执行 即 可 安装 。 

一 个 APK 文件 结构 为 : META-INF\Jar， 此 文件 结构 的 具体 说 明 如 下 所 示 。 

口 “res\”: 存放 资源 文件 的 目录 ; 

口 AndroidManifest.xml: 程序 全 局 配置 文件 ; 
口 classes.dex: Dalvik 字 节 码 ; 

口 resources.arsc: 编译 后 的 二 进 制 资源 文件 。 

Android 在 运行 程序 时 ， 首 先 需要 解压 缩 ,， 这 一 点 和 Symbian 相似 , 而 和 Windows Mobile 
中 的 PE 文件 有 所 区 别 。 这 样 做 程序 的 保密 性 和 可 靠 性 不 是 很 高 ， 通 过 dexdump 命令 可 以 反 
编译 , 但 这 样 做 符合 发 展 规律 , 微软 的 Windows Gadgets 或 者 说 WPF 也 采用 了 这 种 构架 方式 。 
在 Android 平台 中 dalvik vm 的 执行 文件 被 打包 为 APK 格式 ， 最 终 运行 时 加 载 器 会 解压 然后 
获取 编译 后 的 androidmanifest.xml 文件 中 的 permission 分 支 相 关 的 安全 访问 , 但 仍然 存在 很 多 
安全 限制 ， 如 果 你 将 APK 文件 传 到 “/system/app” 文 件 夹 下 会 发 现 执 行 是 不 受 限制 的 。 最 终 
安装 的 文件 可 能 不 放 这 个 文件 夹 中 ， 系 统 的 APK 文件 默认 会 放 入 这 个 文件 夹 ， 它 们 拥有 着 
ROOT 权限 。 
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8.11.2 下 载 APK 应 用 程序 


读者 可 以 从 哪里 获得 好 用 的 Android APK 应 用 程序 ， 并 安装 到 Android 手机 上 呢 ? 对 拥 


者 而 言 ，Android Market 就 是 最 佳 的 地 方 ， 只 要 使 用 手机 内 应 用 程序 列 


有 G1 实体 手机 的 使 用 


二 


表 的 Market 程序 ， 就 


下 载 并 安装 到 G1 手机 J 


可 以 直接 连接 到 Android Market， 而 点 选 喜 爱 的 应 用 程序 后 ， 就 会 直接 
FEF。 不 过 对 使 用 Android 仿真 器 的 使 用 者 而 言 ， 就 没有 如 此 方便 了 ， 


Android 仿真 器 并 没有 Android Market 这 个 应 用 程序 ， 只 能 使 用 内 附 的 浏览 器 浏览 Android 


Market， 为 何 说 是 浏览 呢 ? 


因为 Android Market 不 是 采用 通用 网 页 浏览 方式 来 下 载 文件 ， 虽 


然 可 以 使 用 常见 的 浏览 器 看 到 Android Market 上 的 应 用 程序 ， 但 是 没有 办 法 下 载 到 Android 


仿真 器 或 一 般 的 计算 机 上 ， 原 因 是 Android Market 
式 来 访问 ， 唯 有 通过 内 建 在 G1 手机 内 的 Market 应 用 程序 ， 才 能 下 载 Android Market 网 页 中 


的 应 用 程序 ， 并 自动 安装 到 G1 手机 上 。 


所 以 Android 仿真 器 的 使 ) 


采用 特有 的 网 页 API， 使 用 native UI 的 方 


者 ， 只 好 浏览 该 网 页 上 的 应 用 程序 ， 然 后 通过 搜索 引擎 去 找 


找 看 有 没有 开发 人 员 将 应 用 程序 放 到 Android Market 之 后 , 另外 还 将 APK 文件 放置 在 一 般 网 


页 上 了 。 到 此 为 止 ， 使 用 Android 仿真 器 ， 也 不 需要 这 人 么 灰心 ， 因 为 有 太 多 的 人 遇 到 同样 的 
问题 ， 也 就 生成 非常 多 的 Android 应 用 程序 网 页 ， 您 可 以 浏览 这 些 网 页 并 把 上 面 的 APK 文件 


下 载 到 一 般 计算 机 上 ， 
下 面 列 出 了 常 


CEE 


再 进行 安装 。 


的 APK 应 用 程序 下 载 网 站 。 


http://andappstore.com/ 
http://www.getjar.com/software 
http:/www.phoload.com/android 


http://slideme.org/ 


http://androidforums.com/market/ 


http:/www.cyrket. 


com/ 


http://www.androidfreeware.org/ 
http://androidsoftwaredownload.com/ 
http://www.freeandroidsoft.com/ 


http://code.google. 


com/p/apps-for-android/ 


http://code.google.com/p/openintents/downloads/list 


8.11.3 ”安装 APK 应 用 程序 


影片 播放 软件 时 就 曾经 使 月 


APK 安装 指令 。 


所 有 的 APK 应 用 程序 要 安装 到 Android 仿真 器 上 ， 就 只 有 一 个 指令 ， 就 是 在 3.3 节 安 装 
日 过 的 adb install 指令 , 请 开启 一 个 命令 字符 的 终端 机 窗口 ， 并 


adb install filename.apk 


这 样 adb 指令 就 会 自动 将 flename.apk 应 用 程 


序 安装 到 Android 仿真 器 上 , 而 仿真 器 上 的 


应 用 程序 列表 也 会 立即 出 现 刚 刚 安装 的 应 用 程序 图 标 ， 如 果 应 用 程序 没有 安装 成 功 ， 或 安装 
不 完善 ， 也 可 以 重复 运行 adb install -r filename.apk 指令 ， 这 样 会 保留 已 经 设置 的 信息 ， 而 仪 
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[dll 
ll 


EE 新 安装 应 用 程序 本 身 。 

不 过 在 安装 APK 应 用 程序 组 件 时 ， 不 可 以 同时 运行 多 个 Android 仿真 器 ， 因 为 adb 不 知 
要 将 APK 应 用 程序 安装 到 哪 一 个 仿真 器 ， 最 好 的 方法 就 是 仅 运 行 一 个 Android 仿真 器 。 如 果 
需要 同时 运行 多 个 仿真 器 ， 就 要 在 安装 APK 组 件 时 ， 先 使 用 adb 指定 某 一 个 仿真 器 。 您 可 以 
从 Android 仿真 器 的 窗口 上 ， 看 到 类 似 Android Emulator (5554) 的 字样 ， 而 5554 就 是 仿真 
器 的 运行 序号 ， 每 一 个 仿真 器 有 其 独特 的 运行 序号 ， 只 要 将 adb 加 上 -s <serialNumber> 参数 ， 
就 可 以 将 APK 应 用 程序 安装 到 指定 的 仿真 器 上 。 


| 
征 


adb -s emulator-5554 install filename.apk 
(指定 安装 APK 组 件 在 5554 的 Android 仿真 器 中 


LS 


8.11.4” 移 除 APK 应 用 程序 


如 果 已 经 安装 了 很 多 Android 应 用 程序 ， 如 果 想 要 删除 一 些 应 用 程序 图 标 ， 也 非常 的 简 
单 ， 同 样 是 一 行 指令 就 搞定 了 。adb uninstall 指令 可 以 将 APK 应 用 程序 移 除 。 


adb uninstall package 


例如 ， 


adb uninstall com.android.email (把 email 程序 移 除 ) 


Android 使 用 的 package 名 称 类 似 我 们 浏览 网 页 时 常用 的 域名 方式 ， 所 以 上 面 的 示例 是 将 
email 包 移 除 ， 请 记 住 ， 包 名 称 不 是 您 安装 APK 组 件 时 的 文件 名 或 是 显示 在 Android 仿真 器 
中 的 应 用 程序 名 称 。 另 外 包 名 称 也 并 不 一 定 都 是 com.android 这 样 的 形式 ， 它 可 以 是 各 式 各 样 
的 域名 方式 。 例 如 org.iiiro.iiivpa 或 com.deafcode.android.Cinema。APK 文件 的 包 名 称 完 全 是 
由 当初 的 开发 人 员 所 制定 的 ， 所 以 并 没有 统一 的 命名 方式 ， 唯 一 相同 的 就 是 它 一 定 是 类 似 域 
名 的 命名 格式 。 

另外 ， 在 移 除 该 APK 应 用 程序 时 ， 如 果 想 要 保留 信息 与 缓存 目录 ， 则 加 上 =-k 参数 即 可 。 


adb uninstall -k package ( 移 除 程序 时 ， 保 留 信 息 ) 
不 过 麻烦 的 是 ， 可 能 不 知道 这 个 想 要 移 除 的 应 用 程序 包 名 称 ， 所 以 必须 先 运行 adb shell 
进入 Android 操作 系统 的 指令 列 模式 ， 然 后 到 /data/data 或 /data/app 目录 下 ， 得 知 欲 移 除 的 


包 名称 ， 然 后 使 用 adb uninstall 指令 删除 APK 应 用 程序 ， 这 样 就 可 以 简易 地 从 Android 仿真 
器 将 不 想 使 用 的 APK 应 用 程序 移 除了 。 


~ 


adb shell 
ls /data/data 或 /data/app (查询 包 名 称 ) 
exit 
adb uninstall package ( 移 除 查 询 到 的 包 ) 


平和 运 的 是 ， 从 Android SDK 1.5 版 起 ， 已 经 内 建 应 用 程序 管理 系统 ， 不 需要 再 辛苦 的 使 用 

adb uninstall 指令 移 除 APK 应 用 程序 组 件 ， 只 要 在 Android 手机 主 画 面 点 选 MENU 按键 ， 然 
后 依 序 选择 “Settings” 一 “Applications” 一 “Manage applications” 就 可 以 启动 应 用 程序 管 
画 国 427 


理 系统 。 
用 程序 ， 


8.11.5 


当前 


实现 原理 


本 实例 运行 
Application installer 软 伯 
序 的 URL， 按 后 通过 自 
连接 , 通过 InputStream 将 下 载 文 件 写 入 到 存 作 


Android 开发 完全 实战 宝典 
Android 系统 已 经 安装 
然后 选择 卸载 就 可 以 移 除 该 程 


的 所 有 应 用 程序 都 会 列 出 来 ， 您 上 只 要 选择 想 要 移 


序 了 ， 这 样 就 不 需要 使 用 adb uninstall 指令 。 


后 ， 能 够 远程 下 载 指定 网 址 的 Android 应 |} 
来 安装 这 个 软 人 


o 


打开 文人 
安装 完成 后 
8.11.6 ”具体 实现 


本 实例 的 主 程序 文人 


1) 先 引 ) 


定义 按钮 打开 下 载 程序 


F， 并 根据 文件 扩展 名 ,判断 是 否 为 APK 格式 , 是 则 
， 在 退出 安装 时 通过 方法 delFileO 将 存储 卡 ! 


public class examplel 1 extends Activity 


{ 


private TextView mTextView01; 
private EditText mEditTextO1; 
private Button mButton01; 

private static final String TAG = "DOWNLOADAPK"; 


private String currentFilePath 


= 
. 


private String currentTempFilePath 
private String strURL=""; 
private String fileEx=""; 


private String fileNa=""; 
/** Called when the activity is first created. */ 
(@Override 
public void onCreate(Bundle savedInstanceState) 


{ 


= 
> 


super.onCreate(savedInstanceState); 


setContentView(R.layout.main); 


F。 在 具 


程序 ， 下 载 到 手机 后 打 
本 实现 上 ， 先 设置 一 个 EditText 来 获取 远程 程 
(使 用 java.net 的 URLConnection 对 象 来 创建 
卡 的 缓存 。), 下 载 后 通过 自 定义 方法 openFile() 


月 动 内 置 的 安装 程序 ， 
的 临时 文件 删除 。 


是 examplell.java， 下 面 开始 分 别 介绍 其 实现 流程 。 
] java.io 与 java.net， 然 后 定义 各 个 变量 。 具 体 代码 如 下 。 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 
mButton01 = (Button)findViewById(R.id.myButton!1); 
mbEditText01 =(EditText)findViewById(R.id.myEditText]1); 


2) 定义 setOnClickListener 用 于 监听 按钮 单 击 事件 ， 设 置 文件 下 载 到 本 地 端 ， 取 


装 程序 的 文人 


名称。 


具体 代码 如 下 。 


mButton01.setOnClickListener(new Button.OnClickListener() 


428 国 田 


{ 


public void onClick(View v) 


{ 


始 安装 。 
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放 文件 会 下 载 至 本 地 端 忆 
mTextView01.setText(" 下 载 中 ..."); 
strURL = mEditText01.getText().toString(); 
放 取 得 欲 安装 程序 之 文件 名 称 */ 
fileEx = strURL.substring(strURL .lastIndexOf(".") 
+1,strURL .length()).toLowerCase(); 
fileNa = strURL.substring(strURL.lastIndexOf("/") 
+1,strURL .lastIndexOf(".")); 
getFile(strURL); 
} 

} 

); 


3) 如 果 文 本 框 中 的 远程 地 址 为 室 ， 则 输出 “请 输入 URL” 的 提示 。 具 体 代 码 如 下 。 


mEditText01.setOnClickListener(new EditText.OnClickListener() 


{ 
(@Override 
public void onClick(View arg0) 
{ 
mEditText01.setText(""); 
mTextView01.setText(" 远 程 安装 程序 (请 输入 URD)"); 
) 
); 


4) 定义 方法 getFile(final String strPathb)， 用 于 获取 下 载 的 URL 文件 ， 有 腊 常 则 和 输出 提示 。 
具体 代码 如 下 。 


旋 处 理 下 载 URL 文件 自 定义 方法 */ 
private void getFile(final String strPath) { 
try 
{ 
if (strPath.equals(currentFilePath) ) 
{ 
getDataSource(strPath); 
} 
currentFilePath = strPath; 
Runnabler =new Runnable() 
{ 
public void run() 
{ 
try 
{ 
getDataSource(strPath); 


} 


catch (Exception e) 


T 
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{ 
Log.e(TAG, e.getMessage(), e); 
} 
} 
}; 
new Thread(r).start(); 
} 
catch(Exception e) 
{ 
e.printStack Trace(); 
} 
} 


5) 定义 方法 getDataSource(String strPath)， 用 于 获取 远程 文件 。 流 程 如 下 。 


AAA 


第 一 步 ， 如 果 URL 错误 则 输出 提示 。 

第 二 步 : 通过 myURL 对 象 取得 URL。 

第 三 步 : 创建 连接 对 象 conn。 

第 四 步 : 通过 File 创建 临时 文件 。 

第 五 步 : 取得 暂 存 路 径 ， 并 将 文件 号 入 暂 存 SD 存储 卡 。 
第 六 步 : 通过 openFile(myTempFile) 方 法 打开 文件 进行 安装 。 
LL 体 代 人 码 如 下 。 


放 取 得 远程 文件 */ 
private void getDataSource(String strPath) throws Exception 
{ 
if (IURLUtil.is NetworkUrl(strPath)) 
{ 
mTextView01.setText(" 错 误 的 URL"); 
} 
else 
{ 
放 取 得 URL*/ 
URL myURL = new URL(strPath); 
庶 创 建 连接 %/ 


URLConnection conn = myURL.openConnection(); 


口 
口 
口 
口 
口 
口 


conn.connect(); 
/*InputStream 下 载 文 件 */ 
InputStream is = conn.getInputStream(); 
if (is== null) 
{ 
throw new RuntimeException("stream ls null"); 
} 
诺 创 建 临 时 文件 */ 
File myTempFile = File.create TempFile(fileNa, "."+fileEx); 
旋 取 得 暂 存盘 路 径 * 


es 
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} 
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currentTempFilePath = myTempFile.getAbsolutePath(); 
人 # 将 文件 写 入 暂 存 盘 忆 
FileOutputStream fos = new FileOutputStream(myTempFile); 
byte buff] = new byte[128]; 
do 
{ 

int numread = is.read(buf); 

if (numread <= 0) 

{ 

break; 

} 

fos.write(buf, 0, numread); 
}while (true); 


/# 打 开 文 件 进行 安装 光 
openFile(myTempFile); 
try 
{ 
is.close(); 
} 
catch (Exception ex) 
{ 
Log.e(TAG, "error: " + ex.getMessage(), ex); 
} 


6) 定义 方法 openFile(File 人 ， 设 置 在 手机 上 打开 文件 。 流 程 如 下 。 


Parag 


口 第 一 步 : 调用 getMIMEType0) 方 法 来 取得 文件 类 型 。 
口 第 二 步 : 设置 intent 的 文件 位 置 与 文件 类 型 。 

口 第 三 步 : 判断 文件 类 型 的 方法 ， 并 取得 扩展 名 。 
口 
口 


第 四 步 : 根据 扩展 名 的 类 型 决定 文件 类 型 。 
第 五 步 : 如 果 无 法 直接 打开 则 弹出 软件 列表 ， 供 用 户 选择 。 
具体 代码 如 下 。 


主攻 


手机 上 打开 文件 的 方法 */ 


private void openFile(File f) 


{ 


Intent intent = new Intent(); 
intent.addFlags(Intent.FLAG ACTIVITY NEW_ TASK); 
intent.setAction(android.content.Intent.ACTION VIEW); 


WS 


调用 getMIMEType0 来 取得 MimeType */ 


String type = get MIMEType(D); 


7 


设置 intent 的 file 与 MimeType */ 


网 络 应 用 
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intent.setDataAndType(Uri.fromFile(f),type); 


startActivity(intent); 


上 六 判断 文件 MimeType 的 方法 */ 
private String get MIMEType(File f) 


String type=""; 

String fName=f.getName(); 

席 取得 扩展 名 */ 

String end=fName.substring(fName.lastIndexOf(".") 
+1,fName.length()).toLowerCase(); 


此 依 扩展 名 的 类 型 决定 MimeType */ 
这 end.equals("m4a")llend.equals(C"MP37)llend.equalsC"mid") 
end.equals("xmf')llend.equals("ogg")llend.equals("wav")) 


{ 


type = "audio"; 
} 
else if(end.equals("3gp")llend.equals("mp4")) 
{ 
type = "video"; 
} 


else if(end.equals("jpg")llend.equals("gif')llend.equals("png")|| 
end.equals("jpeg")llend.equals("bmp")) 


{ 
type = "image"; 
} 
else if(end.equals("apk")) 
{ 


/* android.permission.INSTALL PACKAGES */ 
type = "application/vnd.android.package-archive"; 
} 
else 
\ 
type="*"; 
} 
族 如 有 果 无 法 直接 打开 ， 束 跳出 软件 列表 给 用 户 选择 * 
if(end.equals("apk")) 


else 
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{ 
type 二 二 WA 


} 


return type; 


} 


7) 定义 方法 delFile(String strFileName)， 用 于 删除 SD 卡 上 的 临时 文件 。 具 体 代 码 如 下 。 


o 
| 


族 自 定义 删除 文件 方法 */ 
private void delFile(String strFileName) 
{ 
File myFile = new File(strFileName); 
if(myFile.exists()) 
{ 
myFile.delete(); 
} 
} 


8) 定义 方法 onPause0 和 onResume0， 分 别 设置 onPause (暂停 ) 和 onResume (重播 
状态 。 有 具体 代码 如 下 。 


/# 当 Activity 处 于 onPause 状态 时 ,更 改 TextView 文字 状态 */ 
Override 
protected void onPause() 
{ 
mTlextView01 = (TextView)findViewById(R.id.myTextViewl); 
mTextView01.setText(" 下 载 成 功 "); 


un 
a 


super.onPause(); 
} 
诺 当 Activity 处 于 onResume 状态 时 ， 删 除 临 时 文件 浆 
(QOverride 
protected void onResume() 
{ 


雍 删除 临时 文件 六 
delFile(currentTempFilePath); 


super.onResume(); 


} 
} 


执行 后 将 在 文本 框 中 显示 目标 安装 程序 的 路 径 ， 如 图 8-18 所 示 。 实 例 中 的 默认 路 径 是 
http://mz.ruan8.com/soft/2/sougoushoujishurufa 7786.apk, 是 一 个 sogou 输入 法 程序 。 单 击 “ 按 
下 开始 安装 ”按钮 后 ， 开 始 下 载 目 标 文件 ， 如 图 8-19 所 示 。 
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友 面 全 2:23 AM 


局 com.sohu.inputmethod... 


Installing... 


动 剧 全 2:20AM 


http://mz.ruan8.com/soft/2/ 
sougoushoujishurufa_7786.apk 


NNN、S、S、 旭 SU 


按 下 开始 安装 


图 8-18 下 载 目 标 文件 图 8-19 下载 界面 


下 载 完成 后 弹出 安装 界面 ， 如 图 8-20 所 示 。 单 击 图 8-20 中 的 “Install” 按 钮 后 开始 安装 ， 
安装 完成 后 输出 提示 ， 如 图 8-21 所 示 。 


磺 曾 全 2:22 AM 


BS Sogou Input 


SS com.sohu.inputmethod... 


Do you want to install this 
application? 


Application installed 


Allow this application to: 


图 8-20 ”安装 界面 图 8-21 ”安装 完成 界面 


82 U000 3gp00 


观看 在 线 视频 ， 是 当前 智能 手机 的 主要 功能 之 一 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实 
例 的 实现 ， 介 绍 在 Android 中 下 载 观看 3gp 视频 的 基本 流程 。 本 实例 的 源 代码 保存 在 “ 光 
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盘 :daimaNg\example12”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


8.12.1 
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mm 


实现 原理 


视频 一 般 比 较 大 ， 必 须 保证 手机 有 足够 空间 能 够 存储 ， 还 要 确保 下 载 的 视频 能 够 补 
MediaPlayer 所 支持 。 在 本 实例 中 ， 通 过 EditText 来 获取 远程 视频 的 URL， 然 后 将 此 网 址 的 视 
频 下 载 到 手机 的 存储 卡 中 ， 然 后 通过 控制 按钮 来 控制 对 视频 的 处 理 。 在 播放 完毕 并 终止 程序 


后 ， 将 暂 存 到 SD 中 的 临时 视频 删除 。 


8.12.2 具体 实现 
本 实例 的 主 程序 文件 是 example12.java， 下 面 开始 介绍 其 实现 流程 。 


1) 加 载 相关 类 ， 然 后 通过 Activity 实现 SurfaceHolderCallback， 分 别 创建 MediaPlayer 
对 象 和 SurfaceHolder 对 象 。 具 体 代 码 如 下 。 


/* Activity 实现 SurfaceHolder.Callback */ 
public class example12 extends Activity 
implements SurfaceHolder.Callback 
{ 

private TextView mTextView0!1; 

private EditText mEditTextO1; 

/* 创建 MediaPlayer 对 象 */ 

private MediaPlayer mMediaPlayer01; 

旋 用 以 配置 MediaPlayer 的 SurfaceView */ 

private SurfaceView mSurfaceView0!1; 

/* SurfaceHolder 对 象 */ 

private SurfaceHolder mSurfaceHolder01; 


private ImageButton mPlay, mReset, mPause, mStop; 


2) 识别 MediaPlayer 对 象 是 否 已 被 释放 ， 识 别 MediaPlayer 对 象 是 否 正 处 于 和 暂停 状态 ， 并 


用 LogCat 输出 状态 提示 标识 。 具 体 代码 如 下 。 


庆 识别 MediaPlayer 是 否 已 被 释放 */ 
private boolean bISReleased = false; 


/# 识别 MediaPlayer 是 否 正 处 于 暂停 */ 
private boolean bIsPaused = false; 


诬 LogCat 输出 状态 提示 标识 */ 
private static final String TAG = "HippoMediaPlayer"; 


1 
多 


private String currentFilePath = 


private String currentTempFilePath = ""; 
private String strVideoURL = "",; 


3) 设置 播放 视频 的 URL 地 址 ， 用 mSurfaceView01 对 象 绑 定 Layout 上 的 Surface 
View。 然 后 设置 PixnelFormat( 视 频 流 格式 )， 并 设置 SurfaceHolder 为 界面 布局 。 具 体 
代码 如 下 。 
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4) 为 影片 设置 大 小 比例 ， 设 置 4 个 控制 按钮 mPlay、mReset、mPause 和 mStop。 


人 码 如 下 。 


5) 定义 mmPlaysetOnClickListener0 方 法 ， 用 于 播放 监听 处 怕 
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/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


上 # 将 .3gp 图 像 文件 存放 URL 网 址 */ 
strVideoURL = 


"http://new4.sz.3gp2.com//20100205xyy/ 喜 羊 羊 与 灰太狼 %20 踩 高 跷 (www.3gp2.com).3gp'; 


//http://www.dubblogs.ce:8751/Android/Test/Media/3gp/test2.3gp 
mTlextView01 = (TextView)findViewBylId(R.id.myTextView!1); 
mEditText01 = (EditTexbfindViewById(R.id.myEditTextl); 
mEditText01.setText(strVideoURL); 


访 绑 定 Layout 上 的 SurfaceView */ 


mSurfaceView01 = (SurfaceView) fndViewById(R.id.mSurfaceViewl); 


/* 设置 PixnelFormat */ 
getWindow!().setFormat(PixelFormat.TRANSPARENT); 


上 六 设置 SurfaceHolder 为 Layout SurfaceView */ 
mSurfaceHolder01 = mSurfaceView01.getHolder(); 


mSurfaceHolder01.addCallback(this); 


I 


雍 由 于 原 有 的 影片 尺寸 较 小 ， 故 指定 
mSurfaceHolder01.setFixedSize(160, 128); 


为 固定 比例 六 


mSurfaceHolder01.setType(SurfaceHolder.SURFACE TYPE PUSH BUFFERS); 


ImPlay = (ImageButton) fmndViewById(R.id.play); 
mReset = (ImageButton) fmndViewById(R.id.resetb); 
mPause = (ImageButton) findViewBylId(R.id.pause); 
mStop = (ImageButton) fndViewById(R.id.stop); 


上 # 播放 按钮 */ 
mpPlay.setOnClickListener(new ImageButton.OnClickListener() 
{ 


public void onClick(View view) 


{ 
if(checkSDCard()) 


EE， 具 体 代码 如 下 。 


内 体 代 
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{ 
strVideoURL = mEditText01.getText().toString(); 


playVideo(strVideoURL); 
mTextView01.setText(R.string.str_play); 
} 
else 


{ 


ImTextView01.setText(R.string.str_err nosd); 


6) 定义 mReset.setOnClickListener() 方 法 ， 用 于 重新 播放 监听 处 理 。 具 体 代码 如 下 。 


上 重新 播放 按钮 */ 
mReset.setOnClickListener(new ImageButton.OnClickListener() 
{ 
public void onClick(View view) 
{ 
if(checkSDCard()) 
{ 
if(bIsReleased == false) 
{ 
if (mMediaPlayer01 != null) 
{ 


mMediaPlayer01.seekTo(0); 
ImTextView01.SetText(R.string.str_play); 


} 


} 


else 


{ 
mTextView01.setText(R.string.str err nosd); 
} 
} 
); 


7) 定义 mResetsetOnClickListener(0 方 法 ， 用 于 暂停 播放 监听 处 理 。 有 具体 代码 如 下 。 


上 # 暂停 按钮 */ 
mpPause.setOnClickListener(new ImageButton.OnClickListener() 
{ 
public void onClick(View view) 
{ 
if(checkSDCard()) 


{ 
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if (mMediaPlayer01 != null) 
{ 
这 bISReleased == false) 
{ 
if(bIsPaused==false) 
{ 
mMediaPlayer01.pause(); 
blsPaused = true; 


mTextView01.setText(R.string.str pause); 


} 

else if(bIsPaused==true) 

{ 
mMediaPlayer01 .start(); 
blsPaused = false; 


mTextView01.setText(R.string.str_ play); 


} 


else 


{ 


ImTextView01.setText(R.string.str_err nosd); 


) 
} 
1); 


8) 定义 mStop.setOnClickListener() 方 法 ，| 


于 停 趾 


雍 终止 按钮 */ 


上 播放 监听 处 理 


mStop.setOnClickListener(new ImageButton.OnClickListener() 


{ 
public void onClick(View view) 
{ 
if(checkSDCard()) 
{ 
try 
{ 
if (mMediaPlayer01 != null) 
{ 
if(bIsReleased==false) 
{ 
ImMediaPlayer01.seekTo(0); 
ImMediaPlayer01.pause(); 


ImTextView01.setText(R.string.str_stop); 


} 
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具体 代码 如 下 。 
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catch(Exception e) 
{ 
mTextView01.setText(e.toString()); 
Log.e(TAG, e.toString()); 
e.printStackTrace(); 
} 
} 
else 
{ 


ImTextView01.setText(R.string.str_err nosd); 


} 


全 
} 


9) 定义 方法 playVideo0， 用 于 下 载 指定 URL 影片 并 实现 播放 处 理 。 流 程 如 下 。 
口 第 一 步 : 判断 传 入 的 strPath (地址 ) 是否 是 现 有 播放 的 连接 。 

口 第 二 步 : 重新 构建 MediaPlayer 对 象 。 

口 第 三 步 : 设置 播放 音量 。 


所 体 代 码 如 下 。 


入 自 定义 下 载 URL 影片 并 播放 */ 
private void playVideo(final String strPath) 


{ 
try 


{ 
雍 若 传 入 的 strPath 为 现 有 播放 的 连接 ， 则 直接 播放 */ 
if (strPath.equals(currentFilePath) && mMediaPlayer01 != null) 
{ 
ImMediaPlayer01.start(); 
return; 
} 
else if(m MediaPlayer01 != null) 


{ 
ImMediaPlayer01.stop(); 


} 


currentFilePath = strPath; 
/* 重新 构建 MediaPlayer 对 象 */ 
mMediaPlayer01 = new MediaPlayer(); 

庆 设置 播放 音量 */ 
mMediaPlayer01.setAudioStreamType(2); 


/设置 显示 于 SurfaceHolder */ 
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10) 定义 onBufferingUpdate0 方 法 ， 用 于 监听 组 


11) 分 别 ) 


具体 代码 如 下 。 
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mMediaPlayer01.setOnBufferingUpdateListener 
(new MediaPlayer.OnBufferingUpdateListener() 


{ 


mMediaPlayer01.setDisplay(mSurfaceHolder01); 


mMediaPlayer01.setOnErrorListener 
(new MediaPlayer.OnErrorListener() 


{ 


(QOverride 


public boolean onError(MediaPlayer mp, int what, int extra) 


{ 
// TODO Auto-generated method stub 


Log.i 
( 
TAG, 


"Error on Listener, what: " + what + "extra: " + extra 


); 
return false; 


} 


Override 


public void onBufferingUpdate(MediaPlayer mp, int percent) 


{ 


} 


有 


/TODO Auto-generated method stub 
Log.i 
( 
TAG, "Update buffer: "+ 
Integer.toString(percent) + "%" 


~ 


mMediaPlayer01.setOnCompletionListener 
(new MediaPlayer.OnCompletionListener() 


{ 


(QOverride 


public void onCompletion(MediaPlayer mp) 


{ 


/TODO Auto-generated method stub 


过 虑 。 


Log.i(TAG,"mMediaPlayer01 Listener Completed ); 


ImTextView01.setText(R.string.str_done); 


体 代 码 如 下 。 


~ 


setOnPreparedListener() 方 法 和 setOnCompletionListener0 方 法 监听 播放 处 理 


C9 
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} 
D); 
mMediaPlayer01.setOnPreparedListener 
(new MediaPlayer.OnPreparedListener() 
{ 
(@Override 
public void onPrepared(MediaPlayer mp) 
{ 
// TODO Auto-generated method stub 
Log.i(TAG,"Prepared Listener"); 
} 
D); 


12) 定义 方法 run0， 用 于 接受 连接 并 记录 线程 信息 。 首 先 在 线程 运行 中 ， 调 用 自 定义 方 
法 下 载 指定 的 文件 ， 当 下 载 完 后 调用 prepare0 方 法 。 当 有 异常 时 ， 则 输出 错误 信息 。 具 体 代 
人 码 如 下 。 


Runnabler = new Runnable() 
{ 
public void run() 
{ 
try 
{ 
雍 在 线程 运行 中 ， 调 用 自 定义 方法 下 载 指定 文件 */ 
setDataSource(strPath); 
/# 下 载 完 后 才 会 调用 prepare */ 
mMediaPlayer01.prepare(); 
Log.i 


( 
TAG, "Duration: " + mMediaPlayer01.getDuration() 
); 
mMediaPlayer01.start(); 
blIsReleased = false; 
} 
catch (Exception e) 
{ 
Log.e(TAG, e.getMessage(), e); 
} 
} 
} 
new Thread(r).start(); 
} 
catch(Exception e) 
{ 
if (mMediaPlayer01 != null) 


{ 
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} 
13) 定义 方法 setDataSource0， 用 于 线程 启动 的 方式 播放 视频 。 
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/* 


ImMediaPlayer01.stop(); 
mMediaPlayer01.release(); 


自 定义 setDataSource， 由 线程 启动 */ 


private void setDataSource(String strPath) throws Exception 


{ 


if (IURLUtil.isNetworkUrl(strPath)) 


mMediaPlayer01.setDataSource(strPath); 


else 


if(bIsReleased == false) 


{ 

URL myURL = new URL(strPath); 
URLConnection conn = myURL.openConnection(); 
conn.connect(); 
InputStream is = conn.getInputStream(); 
if (is =— null) 
{ 

throw new RuntimeException("stream is null"); 
} 
File myFileTemp = File.createTempFile 
("hippoplayertmp", "."+getFileExtension(strPath)); 


currentTempFilePath = myFileTemp.getAbsolutePath(); 


/*currentTempFilePath = /sdcard/mediaplayert MP39327.dat */ 


FileOutputStream fos = new FileOutputStream(myFileTemp); 
byte bufl| = new byte[128]; 
do 
{ 

int numread = is.read(bu?f); 

if (numread <= 0) 

{ 

break; 

} 

fos.write(buf, 0, numread); 
}while (true); 
mMediaPlayer01.setDataSource(currentTempFilePath); 


L 体 代码 如 下 。 


14) 


15) 


} 
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try 
{ 


is.close(); 


} 


catch (Exception ex) 


{ 
Log.e(TAG, "error: " + ex.getMessage(), ex); 


} 


定义 方法 getFileExtension()， 用 于 获取 视频 的 扩展 名 。 具 体 代 码 如 下 。 


private String getFileExtension(String strFileName) 


{ 


} 


File myFile = new File(strFileName); 

String strFileExtension=myF'ile.getName(); 
strFileExtension=(strFileExtension.substring 
(strFileExtension.lastIndexOf(".")+1)).toLowerCase(); 


if(strFileExtension=="") 

{ 
族 若 无 法 顺利 取得 扩展 名 ， 默 认为 .dat */ 
strFileExtension = "dat"; 


} 


return strFileExtension:; 


定义 方法 checkSDCard0， 用 于 判断 存储 卡 是 否 存 在 。 具 体 代 码 如 下 。 


private boolean checkSDCard() 


{ 


谍 判断 存储 卡 是 否 存 在 */ 
if(android.os.Environment.getExternalStorageState().equals 


(android.os.Environment.MEDIA MOUNTED)) 
{ 


return true; 


} 


else 


{ 


return false; 


(QOverride 
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public void surfaceChanged 
(SurfaceHolder surfaceholder, int format, int w, int h) 
{ 

Log.i(TAG, "Surface Changed"); 
} 


(QOverride 
public void surfaceCreated(SurfaceHolder surfaceholder) 
{ 
Log.i(TAG, "Surface Changed"); 
} 


(QOverride 
public void surfaceDestroyed(SurfaceHolder surfaceholder) 


{ 
Log.i(TAG, "Surface Changed"); 


} 
} 


执行 后 在 文本 框 中 显示 指定 播放 视频 的 URL， 下 载 完 毕 ， 能 实现 播放 处 理 。 如 图 8-22 
所 示 。 


example12 
A A 


//20100205xyy/ 喜 羊 羊 与 灰太狼 120% 
A Ny ~ 
PS 


eel el ele 


(Qald 


Er 


图 8-22 ”执行 效果 
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上 辆 呈 避 三 到 醒 


Android 作为 Google 旗下 的 产品 ， 它 能 够 使 用 官方 的 很 多 强大 的 服务 功能 。 例 如 Google 
API、 日历 相册 和 文件 等 在 本 章 的 内 容 中 , 将 通过 几 个 典型 实例 的 实现 , 来 详细 介绍 Android 
和 其 官方 服务 相 结合 的 基本 知识 和 具体 使 用 流程 。 


生生 轩 员 二 


在 现实 应 用 中 ， 基 于 Google 的 大 多 数 服务 都 需要 通过 官方 进行 的 账号 验证 。 在 本 节 的 内 
容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 通过 账号 验证 来 获取 “Google Account 
Authentication Service ”所 发 出 的 凭据 的 过 程 。 本 实例 的 源 代 码 保存 在 “光盘 :daima 
\9\example1”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


9.1.1 Google Account Authentication Service 介 绍 


当前 的 Google Account 阵容 强大 , 由 单一 的 AuthSub 发 展 到 OAuth Federated Hybird 一 共 
四 种 。 具 体 说 明 如 下 。 
口 AuthSub: 提供 单独 的 google service access， 比 如 gmail contact。 如 果 你 要 一 次 使 用 多 
个 google service， 比 如 同时 需要 读 取 picasaweb 和 youtube 的 数据 ， 那 么 用 authsub 就 
不 那么 合适 了 ， 因 为 你 需要 让 用 户 傻 作 的 登录 google 两 次 。 
口 OAuth: 和 authsub 相 比 ， 提 供 了 sign-on， 一 次 登录 可 以 取得 的 Authtoken 可 以 access 
好 几 个 Google services。 
口 Federated: 本 质 上 是 openid， 他 是 唯一 可 以 让 你 在 使 用 Google Account 登录 自己 网 站 
的 同事 ， 拿 到 用 户 电子 邮件 的 的 方法 。 
口 Hybird: 集成 了 oauth 和 openid, 可 以 让 你 的 网 站 同时 拿 到 Google service 访问 权限 ， 
以 及 用 户 的 电子 邮件 。 
由 此 可 见 。AuthSub 即将 过 时 ， 现 在 的 好 处 是 ， 使 用 Authsub 不 需要 在 Google 注册 自己 
的 网 站 。 


9.1.2 具体 实现 


本 实例 的 主 文件 是 examplel.java、examplel 01 02.java 和 GoogleAuthSub.java， 下 面 
其 实现 流程 。 
1. 文件 example1.java 
examplel.java 文件 的 功能 是 登录 UI 和 获取 用 户 账号 的 密码 ,首先 以 TextView 的 onClickO 
事件 为 起 点 , 调用 自 定 义 的 showLoginForm( 方 法 显示 登录 表单 。examplel.java 文件 的 具体 实 


分 


WW 


别 介 


岂 ”Android 开发 完全 实战 宝典 
现代 码 如 下 。 


public class examplel extends Activity 
{ 

放声 明 变量 */ 

private TextView mTextView0!1; 


private LayoutInflater mInflater0 1; 

private View mView0!1; 

private EditText mEditText01,mEditText02; 
private String TAG = "HIPPO DEBUG"; 
Oe 

private int intShiftPadding = 14; 


/** Called when the activity is first created. */ 
@Override 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


/* 创建 DisplayMetrics 对 象 ， 取 得 屏幕 分 辨 率 */ 
DisplayMetrics dm = new DisplayMetrics(); 
getWindowManager().getDefaultDisplay().get Metrics(dm); 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 


放 将 文字 Label 放 在 屏幕 右上 方 */ 
ImTextView01.setLayoutParams 


( 


new AbsoluteLayout.LayoutParams(intShiftPadding*mTextView01.getText().toString().length(), 
18,(dm.widthPixels-(intShiftPadding*mTextView01.getText().toString().length()))-10,0) 


中 


族 处 理 用 户 单 击 TextView 文字 的 事件 ， 显 示 登 录 对 话 框 */ 
mTextView01.setOnClickListener(new TextView.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 


{ 
// TODO Auto-generated method stub 


入 显示 登录 对 话 框 */ 
ShowLoginForm(); 
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上 # 自 定义 登录 对 话 框 方法 */ 
private void showLoginForm() 
{ 
try 
{ 人族 以 LayoutInflater 取得 主 Activity 的 context */ 
minflater01 = LayoutInflater.from(examplel.this); 
/* 设置 创建 的 View 所 要 使 用 的 Layout Resource */ 
mView01 = mInflater01.inflate(R.layout.login, null); 


/* 账号 EditText */ 
mbEditText01=(EditText)mView01.findViewBylId(R.id.myEditText1); 


/* 密码 EditText */ 
ImEditText02=(EditTexbmView01.fndViewById(R.id.myEditText2); 


/创建 AlertDialog 窗口 来 取得 用 户 账号 密码 */ 
new AlertDialog.Builder(this) 
.setView(mView01) 
.SetPositiveButton("OK", 
new DialogInterface.OnClickListener() 
{ 
/# 覆 盖 onClick0 来 触发 取得 Token 事件 与 完成 登录 事件 */ 
public void onClick(DialogInterface dialog, int whichButton) 
{ 
if(processGoogleLogin(mEditText01.getText().toString(), 
mEditText02.getText().toString())) 
{ 


Intent 1 = new Intent(); 


由 


人 # 登 录 后 调用 注销 程序 (examplel 01 02.java)*/ 
i.setClass(examplel.this, examplel 01 02.class); 
startActivity(D); 

finish(); 


} 
.showO); 


} 


catch(Exception e) 


{ 


e.printStack Trace(); 


} 
/* 调 用 GoogleAuthSub 来 取 的 Google 账号 的 Authentication Token*/ 
private boolean processGoogleLogin(String strUID, String strUPW) 
{ 
try 
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{ 
/* 建 构 自 定义 的 GoogtleAuthSub 对 象 */ 
GoogleAuthSub gas = new GoogleAuthSub(strUID, strUPW); 
人 # 取 得 Google Token*/ 
String strAuth = gas.getAuthSubToken(); 
/* 将 取 回 的 Google Token 写 入 log 中 *%/ 
Log.i(TAG, strAuth); 


} 


catch (Exception e) 


{ 


e.printStack Trace(); 
} 
return true; 
} 
} 


showLoginForm() 方 法 是 一 个 使 用 LayoutInflater 获取 主 Activity (活动 程序 ) 的 关联 程序 ， 
搭配 AlertDialog〈 提 示 框 所 构建 的 登录 表单 。 当 用 户 输入 账号 和 密码 后 ， 开 始 重 ? 
DialogInterface.OnClickListener() 的 onClickO 事 件 来 调用 自 定义 的 processGoogleLogin0 方 法 处 
时 和 Google 账号 验证 的 连接 事件 。 当 通过 Google 验证 后 取得 Google Authentication Token ( 令 
牌 )， 通 过 Intent 打开 examplel 01 02.java 以 改变 UI 的 状态 。 

2. 文件 example1_01_02.java 

examplel_01_02.java 文件 的 功能 是 注销 并 返回 登录 界面 。 即 将 原来 的 登录 状态 ， 改 为 注 
销 状 态 ， 并 实现 TextView 的 onClick0 方 法 。 当 用 户 单 击 TextView， 则 通过 自 定义 的 Intent 来 
调用 example.java， 返 回 到 程序 的 等 待 状态 。 文 件 examplel 01 02.java 的 具体 实现 代码 如 下 。 


3 


De 


public class examplel 01 02 extends Activity 
{ 

private TextView mTextView03; 

大 时 严 凶 鸭 国 十 区 

private int intShiftPadding = 14; 


(QOverride 
protected void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.loginok); 


匀 创建 DisplayMetrics 对 象 ， 取 得 屏幕 分 辨 率 */ 
DisplayMetrics dm = new DisplayMetrics(); 
getWindowManager().getDefaultDisplay().get Metrics(dm); 


/* 通 过 findViewById0 来 取得 TextView 对 象 */ 
mTextView03 = (TextView)findViewByld(R.id.myTextView3); 
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庆 将 文字 Label 放 在 屏幕 右上 方 */ 
mTextView03.setLayoutParams 
( 
new 
AbsoluteLayout.LayoutParams(intShiftPadding*mTextView03.getText().toString().length(),18,(dm.widthPixels-(int 
ShiftPadding*mTextView03.getText().toString().length()))-10,0) 
); 


/* 处 理 用 户 单 击 TextView 文字 的 事件 处 理 *%/ 
mTextView03.setOnClickListener(new TextView.OnClickListener() 
{ 
/# 履 盖 onClick0 事 件 */ 
(QOverride 
public void onClick(View v) 
{ 
Intent 1= new Intent(); 
/# 注 销 后 调用 登录 程序 (examplel_ 01 02.java)*/ 
i.setClass(examplel 01 02.this, examplel.class); 
startActivity(i); 
finish(); 


3. 文件 GoogleAuthSub.java 

GoogleAuthSub.java 的 功能 是 通过 HttpGet 传输 网 络 数据 ， 并 获取 Google 的 日 历 服 务 
Token。 此 文件 是 整个 实例 的 核心 ， 通 过 Google 提供 的 ClientLogin 机 制 ， 使 用 HttpPost 连接 
到 https:/www.google.com/accounts/ClientLogin， 并 同时 将 用 户 账号 和 密码 及 其 相关 参数 以 
Name Value Pair 字符 串 带 入 ， 通 过 自 定 义 的 getAuth() 方 法 获取 Google 认证 的 Authentication 
Token， 然 后 模拟 Google 网 络 服务 的 的 AuthSub 的 方法 ， 将 自 定 义 的 Header0 和 HttpGet 方法 
传 入 Token 来 获取 用 户 Google Calendar 服务 中 的 所 有 日 历数 据 ， 并 以 XML 文件 存储 在 临时 
文件 中 ， 作 为 使 用 Google 服务 的 规范 。 文 件 GoogleAuthSub.java 的 具体 实现 代码 如 下 。 


public class GoogleAuthSub 

{ 
放声 明 变 量 */ 
private DefaultHttpClient httpclient; 
private HttpPost httpost; 
private HttpResponse response; 
private String strGoogleAccount; 
private String strGooglePassword,; 
private String TAG = "IRDC DEBUG'"; 
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/*GoogleAuthSub 对 象 的 构造 器 */ 
public GoogleAuthSub(String strUID, String sttPWD) 
{ 
this.strGoogleAccount = strUID; 
this.strGooglePassword = sttrPWD; 
httpclient = new DefaultHttpClient(); 
httpost = new HttpPost("https://www.google.com/accounts/ClientLogin"); 


放 定 义 取得 Google Token 方法 */ 
public String getAuthSubToken() 
{ 
/创建 Name Value Pair 格式 的 字符 串 */ 
List <NameValuePair> nvps = new ArrayList <NameValuePair>(); 


nvps.add(new BasicNameValuePair("Email", this.strGoogleAccount)); 
nvps.add(new BasicNameValuePair("Passwd", this.strGooglePassword)); 
nvps.add(new BasicNameValuePair("source", "MyApiV1")); 
nvps.add(new BasicNameValuePair("service", "cl")); 
String GoogleLoginAuth=""; 
try 
{ 
/* 创 建 Http Post 连接 */ 
httpost.setEntity(new UrlEncodedFormEntity(nvps, HTTP.DEFAULT CONTENT_CHARSET)); 
response = httpclient.execute(httpost); 
if( response.getStatusLine().getStatusCode()!=200 ) 
{ 


return  ; 


} 

访 取 回 Google Token*/ 

InputStream is = response.getEntity().getContent(); 
GoogleLoginAuth = getAuth(is); 


/模拟 HTTP Header*/ 
Header[] headers = new BasicHeader[6]; 


1 1 


headers[0] = new BasicHeader("Content-type", "application/x-www-form-urlencoded"); 
headers[1] = new BasicHeader("Authorization", "GoogleLogin auth=\""+GoogleLoginAutht™""); 
headers[2] = new BasicHeader("User-Agent", "Java/1.5.0 06"); 

headers[3] = new BasicHeader("Accept", "text/html, image/gif, image/jpeg, *; qd=.2, */*; q=.2"); 
headers[4| = new BasicHeader("Connection", "keep-alive"); 

/* 发 出 Http Get 请 求 登录 Google Calendar 服务 作 范 例 */ 

HttpGet httpget; 

String feedUrl2 = "http://www.google.com/calendar/feeds/default/allcalendars/full"; 

httpget = new HttpGet(feedUrl2); 


httpget.addHeader(headers[0]); 


httpget.addHeader(headers[1]); 
httpget.addHeader(headers[2]); 
httpget.addHeader(headers[3]); 
httpget.addHeader(headers[4]); 

/# 取 得 Google Calendar 服务 应 答 */ 

response = httpclient.execute(httpget); 

String strTemp01 = convertStreamToString(response.getEntity().getContent()); 
Log.i(TAG, strTemp01); 

/指定 暂 存盘 位 置 %% 

String strEarthLog = "/sdcard/googleauth.log"; 

Buffered Writer bw; 

bw= new BufferedWriter (new FileWriter(strEarthLog)); 
放 将 取 回 文件 写 入 暂 存 盘 中 % 

bw.write( strTemp01, 0, strTemp01.length() ); 
bw.flush(); 


} 


catch (UnsupportedEncodingException e) 


{ 


e.printStack Trace(); 


} 


catch (ClientProtocolException e) 


{ 


e.printStack Trace(); 


} 
catch (IOException e) 


{ 


e.printStack Trace(); 


} 


catch(Exception e) 


{ 


e.printStack Trace(); 


} 
return GoogleLoginAuth:; 


人/# 自 定义 读 取 token 内 容 的 方法 */ 


public String getAuth(InputStream is) 
{ 
BufferedReader reader = new BufferedReader(new InputStreamReader(is)); 
String line = null; 
String strAuth=""; 
try 
{ 
while ((line = reader.readLine()) != null) 


{ 
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Log.d(TAG, ": "+line); 
if( line.startsWith("Auth=")) 
{ 
strAuth=line.substring(5); 
Log.i("auth",": "+strAuth); 


} 
catch (IOException e) 
{ 

e.printStack Trace(); 
} 
finally 
{ 

try 

{ 

is.close(); 

} 

catch (IOException e) 

{ 


e.printStack Trace(); 


} 
return strAuth:; 
} 
记 将 数据 转 为 字符 串 方 法 */ 
public String convertStreamToString(InputStream 1s) 
{ 


BufferedReader reader = new BufferedReader(new InputStreamReader(is)); 


StringBuilder sb = new StringBuilder(); 
String line = null; 
try 
{ 
while ((line = reader.readLine()) != null) 


{ 
sb.append(line); 


} 
catch (IOException e) 


{ 


e.printStack Trace(); 


} 
finally 


{ 
try 
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is.close(); 
} 
catch (IOException e) 
{ 
e.printStack Trace(); 
} 
} 
return sb.toString(); 
} 
} 


执行 后 的 效果 如 图 9-1 所 示 ， 单 市 “登录 ”链接 后 显示 登录 表单 界面 ， 如 图 9-2 所 示 ; 
输入 账号 和 密码 并 单 击 “OK” 按 钮 后 弹出 成 功 获取 Token 提示 。 如 图 9-3 所 示 。 


登录 


.一 国史 国外 3:52w ss ] 
example1 一 
输入 密码 
图 9-1 执行 效果 图 9-2 登录 表单 


吗 园地 贱 售 3:55 Anv 


example1.01 .02 


成 功 获 取 了 Tokenll 


图 9-3 成 功 获取 Token 提示 
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当今 社会 ， 信 息 已 经 成 为 第 一 生产 力 。 在 巨大 的 网 络 资源 中 ， 检 索 站 点 过 勃发 展 ， 百 度 、 
雅虎 和 Google, 都 已 经 站 在 了 时 代 的 最 前 沿 。 在 Android 官方 服务 中 ,提供 了 Google Search API 
实现 检索 处 理 。 在 本 节 的 内 容 中 , 将 通过 一 个 具体 实例 的 实现 , 介绍 在 Android 中 通过 Google 
Search API 实现 检索 处 理 的 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\9\example2”， 下 面 开 
台 讲解 本 实例 的 具体 实现 流程 。 
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9.2.1 Google Search API 的 使 用 流程 


使 用 Google Search API 的 基本 流程 如 下 。 

第 一 步 : 构造 一 个 搜索 服务 “容器 ”。google.search.SearchControl 的 实例 代表 页 面 上 的 一 
个 搜索 控件 ， 这 个 控件 是 多 种 “搜索 服务 ”的 “容器 ”。 

第 二 步 : 构造 一 个 搜索 “服务” 对象。 构造 方法 google.search.LocalSearch() 可 以 构造 一 个 
“搜索 服务 ”对 象 。 尽 管 默 认 的 是 有 一 个 搜索 框 和 结果 列表 ,但 这 些 都 是 可 以 改变 的 ， 甚 至 搜 
索 结 果 的 内 容 都 是 可 以 改变 的 。 
第 三 步 : 向 容 器 中 添加 “搜索 服务 ”。 此 功能 通过 方法 searchControl.addSearcher(searcher, 
options) 实 现 ， 其 中 参数 options 的 类 型 为 google.search.Search- Options。 在 此 可 以 添加 的 搜索 
服务 有 多 种 ， 到 目前 为 止 Google 公司 提供 了 以 下 类 型 的 搜索 器 。 


* google.search.LocalSearch 
* google.search. WebSearch 

* google.search.VideoSearch 
* google.search.BlogSearch 

* google.search. NewsSearch 
* google.search.ImageSearch 
* google.search.BookSearch 
* google.search.PatentSearch 


第 四 步 : SearchControl 对 象 调用 draw 方法 ， 按 照 drawOptions 参数 画 出 搜索 框 。 
searchControl.draw(document.getElementByld("from"), drawOptions) 
上 述 步 又 是 一 般 的 流程 , 但 Google 提供 了 很 多 选项 来 定制 这 些 服务 。 主 要 的 选项 有 两 种 ， 
一 种 是 SearcherOptions， 男 一 种 是 DrawOptions 。 


9.2.2 具体 实现 
本 实例 的 主 文件 是 example2.java 和 MyAdapter.java， 下 面 分 别 介 绍 其 实现 流程 。 
1. 文件 example2.java 
文件 example2.java 功能 是 创建 MyAdapter 对 象 ,此 对 象 是 自己 实现 的 BaseMyAdapter 类 。 
文件 example2.java 的 具体 实现 代码 如 下 。 


I 


package irdc.example2; 


import irdc.example2.R; 

import android.app.Activity; 

import android.os.Bundle; 

import android.widget.AutoCompleteTextView; 


public class example2 extends Activity 


{ 


private AutoCompleteTextView myAutoCompleteTextView1l; 
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/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 
myAutoCompleteTextViewl = (AutoCompleteTextView) findViewById(R.id.myAutoComplete- 
TextView!1); 


/#new 一 个 自己 实现 的 BaseAdapter */ 
MyAdapter adapter = new MyAdapter(this); 
myAutoCompleteTextView!1.setAdapter(adapter); 


} 


2. 文件 MyAdapter.java 

MyAdapter 继承 于 BaseAdapter， 可 以 通过 宪 盖 Filterable 对 象 中 的 getFilter() 方 法 来 对 输 
入 的 关键 字 进 行动 态 处 理 。 当 用 户 输入 关键 字 时 ，performFiltering0 方 法 所 返回 的 FilterResults 
《过 滤 结 果 ) 就 是 查询 后 的 结果 。 文 件 MyAdapterjava 的 具体 实现 代码 如 下 。 


public class MyAdapter extends BaseAdapter Implements Filterable 


{ 
ArrayList<String> keyWordValue = new ArrayList<String>(); 
ArrayList<String> resultValue = new ArrayList<String>(); 
private Context mContext; 
LinearLayout.LayoutParams param!; 


public MyAdapter(Context context) 
{ 


mContext = context; 


paraml =new LinearLayout.LayoutParams( 
LinearLayout.LayoutParams.WRAP CONTENT, 
LinearLayout.LayoutParams.WRAP CONTENT); 


(QOverride 
public int getCount() 


{ 
return keyWordValue.size(); 


(QOverride 
public Object getItem(int position) 
{ 
return keyWordValue.get(position); 
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} 


(QOverride 


public long getItemId(int position) 


{ 


return position; 


(QOverride 


public View getView(int position, View view, ViewGroup viewGroup) 


{ 


LinearLayout myLinearLayout = new LinearLayout(mContext); 
myLinearLayout.setOrientation(LinearLayout.HORIZONTAL); 


if (position >= keyWordValue.size()) 
return myLinearLayout; 
上 第 一 个 TextView 存放 关键 字 */ 
TextView keyWordTextView = new TextView(this.mContext); 


keyWordTextView.setTextColor(mContext.getResources().getColor( 
R.drawable.blue)); 
keyWordTextView.setTextSize(18); 
keyWordTextView.setWidth(180); 
try 
{ 
keyWordTextView 
.setText(keyWordValue.get(position).toString()); 
} catch (java.lang.IndexOutOfBoundsException i) 
{ 
keyWordTextView.setText(""); 
} 
族 第 二 个 TextView 存放 关键 字 结 果 数 量 */ 
TextView resultTextView = new TextView(this.mContext); 


resultTextView.setTextColor(mContext.getResources().getColor( 
R.drawable.red)); 
resultTextView.setTextSize(18); 
try 
{ 
resultTextView.setText(resultValue.get(position).toString()); 
} catch (java.lang.IndexOutOfBoundsException 1i) 
{ 
resultTextView.setText("™"); 
} 
myLinearLayout.addView(keyWordTextView, param!1); 
myLinearLayout.addView(resultTextView, paraml); 


return myLinearLayout; 
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(QOverride 

public Filter getFilter() 

{ 
/IODO Auto-generated method stub 
Filter myFilter = new Filter() 


{ 


(QOverride 
protected FilterResults performFiltering(CharSequence text) 


{ 


FilterResults fr = new FilterResults(); 
keyWordValue = new java.util.ArrayList<String>(); 
resultValue = new java.util. ArrayList<String>(); 
if (text =— null || text.length() == 0) 
{ 

fr.count = keyWordValue.size(); 

fr.values = keyWordValue; 

return fr; 


/* 输入 关键 字 后 调用 Google 关键 字 API */ 
changeResult(getGoogleAPI(text.toString())); 


fr.count = keyWordValue.size(); 
fr.values = keyWordValue; 
return fr; 


(QOverride 
protected void publishResults(CharSequence text, 
FilterResults filterResults) 
{ 
if (filterResults != null && filterResults.count > 0) 
notifyDataSetChanged(); 
else 
notifyDataSetInvalidated(); 


} 
上 
return myFilter; 


} 


dD 


/ 


/# 访问 Google-API 取得 返回 的 结果 字符 串 
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private String getGoogleAPI(String text) 
{ 
String uri=""; 
try 
{ 
人 # 输入 的 字 要 encode */ 
uri= "http://www.google.com/complete/" 
+ "search?hl=en&js=true&qu=" 
+ URLEncoder.encode(text, "utf-8"); 
} catch (UnsupportedEncodingException el) 
| 


// TODO Auto-generated catch block 
el.printStackTrace(); 


URL googleUrl = null; 
HttpURLConnection conn = null; 
InputStream is = null; 
BufferedReader in = null; 
String resultStr = ""; 
入 觅 合生 谍 光 
try 
{ 

googleUrl = new URL(uri); 

局 困 济 过 渗 椰 

conn = (HttpURLConnection) googleUrl.openConnection(); 


int code = conn.getResponseCode(); 

族 连接 OK 时 */ 

if (code == HttpURLConnection.HTTP_ OK) 
{ 


旋 取得 返回 的 InputStream */ 
is = conn.getInputStream(); 


in=new BufferedReader(new InputStreamReader(is)); 
String inputLine; 


旋 一 伍 一 得 这 豚 椰 
while ((inputLine = in.readLine()) != null) 


{ 
resultStr += inputLine; 
} 
} 
} catch (IOException e) 
{ 


// TODO Auto-generated catch block 
458 国 国 


} 
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e.printStack Trace(); 
} finally 
{ 
try 
{ 
if (is != null) 
is.close(); 
if (conn != null) 
conn.disconnect(); 
} catch (Exception e) 


{ 


} 
} 


return resultStr; 


} 


上 处 理 返 回 的 字符 串 变 成 ArrayList */ 
private void changeResult(String text) 


{ 


String resultStr = ""; 
String startSub = "new Array(2, "; 
String endSub = "), new Array"; 
int start = text.indexOf(start Sub); 
int end = text.indexOf(endSub); 
if (start l= -1 && end != -1) 
{ 
resultStr = text.substring(start + startSub.length(), end); 
从 去 掉 前 后 的 "*/ 
resultStr = resultStr.substring(1, resultStr.length() - 1); 
6#* 以 “,， ”来 分 隔 字符 串 变 成 字符 串 数组 q 
String total[] = resultStr.split(™\\", \""); 
for (inti= 0;1< total.length / 2; i++) 
{ 
keyWordValue.add(total[i * 2]); 
/* 将 results 字符 串 去 掉头 
resultValue 
.add(total[i * 2 + 1].replaceAll(" results", "")); 


号 过 
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在 上 述 代 码 中 ， 在 MyAdapter 类 中 重 写 了 getView0 方 法 ， 在 其 中 放 了 2 个 TextView， 一 


字 ， 男 一 个 显示 结果 数量 。 因 为 AutoCompleteTextView 组 件 绑 定 了 MyAdapter， 
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所 以 当 Adapter 动态 向 Google 获取 查询 结果 时 ， 也 顺便 更 新 了 AutoCompleteTextView 下 拉 荣 
单 里 的 结果 。 
执行 后 的 效果 如 图 9-4 所 示 。 


六 5554 ss 


[= 园区 间 生 5:42m 
鱼 


example2 


ul 
站 / (Rg A 
Weay 


pa 


9-4 执行 效果 


9.3” Google Chart API 生成 二 维 条 码 


通过 Google 公司 提供 的 Google Chart API， 可 以 方便 的 生成 动态 二 维 条 码 。 这 样 开 发 人 
员 无 需 掌握 GD Libray 等 知识 ， 降 低 了 开发 门 榴 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 
实现 , 介绍 在 Android 中 通过 Google Chart API 动态 生成 二 维 条 码 的 具体 流程 。 本 实例 的 源 代 
码 保存 在 “光盘 \daima\9\example3”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


9.3.1 _ Google Chart API 基础 

Google Chart API 为 每 个 请 求 返 回 一 个 PNG 格式 图 片 。 目前 提供 如 下 类 型 图 表 : 折线 图 、 
柱状 图 、 饼 图 、 维 恩 图 和 散 点 图 。 可 以 设 定 图 表 尺 寸 、 颜 色 和 图 例 。 可 以 在 网 页 中 使 用 <img> 
元 素 插 入 图 表 ， 当 浏览 器 打开 该 网 页 时 ，Chart API 提供 即时 图 表 。 

所 有 Chart API URL 都 应 使 用 如 下 格式 。 


http://chart.apis.google.com/chart?<parameter 1>&<parameter 2>&<parameter n> 
注意 : 每 个 URL 所 有 字符 必须 在 同一 行内 。 


多 个 参数 间 使 用 & 作 为 分 隔 符 ， 可 以 使 用 任意 多 个 参数 ， 例 如 图 9-5 所 示 。 


50 Kb 


Mar APr May June July 


9-5 


http://chart.apis.google.cony/chart?cht=lc&chs=200x125&chd=s:hello World&chxt=x,y&chxl= 
0:[MarlAprIMaylJunelJuly|1:||S0+Kb 
各 个 参数 的 具体 说 明 如 下 。 
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口 http://chart.apis.google.com/chart?: Chart API 的 调用 地 址 。 
口 上 :参数 分 隔 符 。 

口 chs=200x125: 图 表 尺 寸 。 

口 chd=s:helloWorld: 图 表 数 据 值 。 

口 cht=lc: 图 表 类 型 。 

口 chxt=x,y: 显示 x、y 轴 坐 标 。 

口 chxl=0:|IMar|AprjMay|JunelJuly|1:|50+Kb: x、y 轴 坐 标 值 。 
可 以 在 网 页 中 使 用 img 元 素 搬 入 图 表 ， 例 如 ， 


<img src="http://chart.apis.google.com/chart?chs=200x125&amp;chd=s:helloWorld&amp;cht=lc&amp; 
chxt=x,y&amp;chxl=0:|IMar|lAprIMay|JunelJuly|1:||50+Kb" alt="Sample chart" /> 


注意 : 在 HTMLimg 元 素 中 ，URL 属性 中 作 字 符 应 书写 为 转 义 字符 &amp;。 


所 有 请 求 必须 包含 以 下 参数 。 

口 图 表 尺寸 。 

口 图 表 数 据 。 

口 图 表 类 型 。 

其 他 参数 均 为 可 选 参数 ， 各 类 型 图 表 有 效 参数 如 表 9-1 所 示 。 


表 9-1 有 效 参数 


参数 


赫 
将 


亚 
I 


[3 

到 
入 
之 
入 
DS 
法 


数据 颜色 

区 域 、 背 景 填充 
数值 缩放 
线性 过 渡 填 充 
线性 条 纹 填充 
图 表 图 例 
多 轴 标 注 

网 格 线 
多 状 标记 
水 平 区 间 填 充 


| 人 | 


| 


垂直 区 间 填 充 
折线 样式 
数据 区 块 填充 
柱 形 、 间 隔 宽度 
柱状 图 基准 线 
饼 图 、 仪 表 图 标注 v v 


Rs | | Se | ph) | we. | | | | 
| | 


i | Es ES) RAR 
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9.3.2 具体 实现 

本 实例 的 主 文件 是 example3.java 和 main.xml， 下 面 分 别 介绍 其 实现 流程 。 
在 文件 example3.java 中 ， 通 过 自 定义 方法 genGoogleQRChart() 获 取 要 显示 远程 图 像 的 网 
址 ， 然 后 以 “<img src=”>” 的 方式 来 组 成 HTML 标记 。 在 具体 成 像 时 ， 通 过 了 WebView 组 
件 来 显示 HTML 的 内 容 。 文 件 example3.java 的 具体 实现 代码 如 下 。 


public class example3 extends Activity 
{ 
private Button mButton01; 
private EditText mEditTextO1; 
private WebView mWebView0!1; 
private boolean bInternetConnectivity=false; 


/** Called when the activity is first created. */ 
(QOverride 
public void onCreate(Bundle savedInstanceState) 
{ 
super.onCreate(savedInstanceState); 
setContentView(R.layout.main); 


访 测试 手机 是 否 具 有 连接 Google API 的 连接 能 力 */ 
if(checkInternetConnection 
("http://code.google.com/intl/zh-T W/apis/chart/","utf-8") 
) 
{ 


bInternetConnectivity = true; 


} 


mWebView01 = (WebView)findViewById(R.id.myWebView!); 
mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 
Override 
public void onClick(View v) 
{ 
// TODO Auto-generated method stub 
if(mEditText01.getText().toString()!="" && 
bInternetConnectivity==true) 


{ 
mWebView01.loadData 
( 
族 调用 自 定义 云端 生成 条 码 的 方法 */ 
genGoogleQRChart 
( 
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mEditText01.getText().toString(),120 
),"text/html", "utf-8" 


mEditText01 = (EditTexbfindViewById(R.id.myEditTextl); 
mEditText01.setText(R.string.str text); 


mEditText01.setOnKeyListener(new EditText.OnKeyListener() 
{ 

(QOverride 

public boolean onKey(View v, int keyCode, KeyEvent event) 


{ 


if(mEditText01.getText().toString(O)!="" && 
bInternetConnectivity==true) 
{ 
mWebView01.loadData 
( 
genGoogleQRChart 
( 
mEditText01.getText().toString(),120 
),"text/html", "utf-8" 
); 
} 


return false; 


2 
} 


族 调用 Google API， 产 生 二 维 条 形 码 */ 
public String genGoogleQRChart(String strToQRCode, intstrWidth) 
{ 
String strReturn=""; 
try 
{ 
strReturn = new String(strToQRCode.getBytes("utf-8")); 


/ 


Ud 


/* 组 成 Google API 需要 的 传输 参数 字符 上 
strReturn = "<html><body>"+ 

"<img src=http://chart.apis.google.com/chart?chs="+ 
strWidth+"x"+strWidth+"&chl="+ 
URLEncoder.encode(strReturn, "utf-8")+ 


"&choe=UTF-8&cht=qr></body></html>"; 
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catch (Exception e) 


{ 


e.printStack Trace(); 


} 


return strReturn; 


入 检查 网 络 连接 是 否 正常 */ 
public boolean checkInternetConnection 
(String strURL, String strEncoding) 
{ 
族 最 多 延 时 n 秒 ， 若 无 应 答 则 表示 无 法 连接 */ 
int intTimeout = 5; 
try 
{ 
HttpURLConnection urlConnection= null; 
URL url = new URL(strURL); 
urlConnection=(HttpURLConnection)url.openConnection(); 
urlConnection.setRequestMethod("GET"); 
urlConnection.setDoOutput(true); 


urlConnection.setDoInput(true); 
urlConnection.setRequestProperty 
( 

"User-Agent"," Mozilla/4.0"+ 

" (compatible; MSIE 6.0; Windows 2000)" 


); 


urlConnection.setRequestProperty 
("Content-type", "text/html; charset="+strEncoding); 
urlConnection.setConnectTimeout(1000*intTimeout); 
urlConnection.connect(); 
if (urlConnection.getResponseCode() == 200) 
{ 
return true; 
} 


else 


{ 


return false; 


} 
} 


catch (Exception e) 


{ 


e.printStack Trace(); 
return false; 
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上 # 自 定义 BIG5 转 UTF-8*/ 
public String big52unicode(String strBIGS) 
{ 
String strReturn=""; 
try 
{ 
strReturn = new String(strBIGS.getBytes("big5"), "UTF-8"); 
} 
catch (Exception e) 
{ 
e.printStack Trace(); 
} 
return strReturn; 


} 


族 自 定义 UTF-8 转 BIG5 */ 
public String unicode2big5 (String strUTFS) 
{ 
String strReturn=""; 
try 
{ 
strReturn = new String(strUTFS8.getBytes("UTF-8"), "bigS"); 
} 
catch (Exception e) 
{ 
e.printStack Trace(); 
} 
return strReturn; 
} 
} 


执行 后 的 效果 如 图 9-6 所 示 。 


吗 国友 阐 便 6:31ww 
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图 9-6 执行 效果 
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9.4 Google DOOUODOOD 


时 候 就 推出 


Google 公司 在 很 早 的 
实现 ， 介 


了 地 图 应 用 。 在 


在 Android 中 通过 Google MapView (谷歌 


本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 
也 图 ) 和 GeoPoint 〈 表 示 经 度 和 纬度 的 


类 ) 实现 地 图 经 纬 应 用 的 流程 。 本 实例 的 源 代码 保存 在 “光盘 :\daima\9\example4”， 下 面 开始 


讲解 本 实例 的 具体 实现 流程 。 


9.4.1 Google MapView 基 础 


Android 支持 GPS 和 网 络 地 图 ， 
置 的 月 


及 务 (Location Based Service ) 的 简 


\ 证 


GSM 


或 大 地 坐标 )， 在 地 理 信 息 系统 (Geographic Information System，GIS) 平台 的 支持 下 ， 为 用 


户 提供 相应 服务 的 一 种 增值 业务 。 
Android 文 持 地 理 定位 
应 用 程序 可 以 定时 请 求 更 新 设备 当前 
器 来 实现 如 下 功能 : 
言 息 。 在 下 面 的 内 容 中 ， 
1. android.location 的 功能 类 


(1) Android Location API 


服务 的 API。 该 地 理 定位 月 


通常 将 各 种 不 同 的 定位 技术 成 为 LBS。LBS 是 基于 位 
称 , 它 是 通过 电信 移动 运营 商 的 无 线 电 通讯 网 络 (如 
网 、CDMA 网 ) 或 外 部 定位 方式 (如 GPS) 获取 移动 终端 用 户 的 位 置信 息 〈 地 理 坐 标 


有 务 可 以 用 来 获取 当前 设备 的 地 理 位 置 。 


的 地 理 定位 信息 


以 经 纬度 和 半径 划 定 的 一 个 区 域 
开始 讲解 android.location 中 和 定位 有 关 的 功 


以 下 是 包 中 关于 定位 功能 的 比较 重要 的 类 。 


口 LocationManager: 本 类 提供 


口 LocationProvider: 该 类 是 定位 提供 


能 够 通过 在 LocationProvider 


位 置 的 功能 。 

口 LocationListener: 提供 定位 信息 发 生 改 变 时 的 
册 监 听 器 对 象 。 

口 Criteria: 该 类 使 得 应 用 
提供 者 。 

(2) Map API 


Android 也 提供 了 一 组 访问 地 图 


下 
\/. 让 


户 当 前 的 地 理 


‘ 心 \o 


者 的 抽象 类 。 


， 当 设备 出 入 该 


应 用 程序 也 可 以 借助 一 个 Intent 接收 
区 域 时 ， 可 以 发 出 提醒 


访问 定位 服务 的 功能 ， 也 提供 获取 最 佳 定位 提供 者 的 功 
能 。 另 外 ， 临 近 和 警报 功能 也 可 以 借助 该 类 来 实现 。 


定位 提供 者 具备 周期 性 报告 设备 地 理 


E 


回调 功能 。 必 须 寻 


E 在 定位 管理 


到 
~ 


器 中 汾 


PP 设置 的 属性 来 选择 合适 的 定位 


的 API, 借助 Map 及 定位 API, 就 能 在 地 图 上 显示 用 
在 Android 中 定义 了 一 个 名 为 com.google.android.maps 的 包 ， 其 中 


| 


包含 了 一 系列 用 于 在 Map 上 显示 、 控 制 和 层 登 信息 的 功能 类 ， 以 下 是 该 包 中 最 重要 的 几 
个 类 
从 。 
口 MapActivity: 这 个 类 是 用 于 显示 MAP 的 Activity 类 ， 它 需要 连接 底层 网 络 。 
口 MapView: MapView 是 用 于 显示 地 图 的 View 组 件 ， 它 必须 和 MapActivity 配合 使 用 。 
口 MapController: MapController 用 于 控制 地 图 的 移动 。 
口 Overlay: 这 是 一 个 可 显示 于 地 图 之 上 的 可 绘制 的 对 象 。 


口 GeoPoint: 一 个 包含 经 纬度 位 置 的 对 象 。 


466 看 看 


第 9 章 _ 绑 定 官方 的 服务 
2. Android 定 位 的 基本 流程 
了 解 了 Android 中 和 定位 处 理 相 关 的 类 后 , 下 面 将 简要 介绍 在 Android 中 实现 定位 处 理 的 
(1) 准备 Activity 类 
目标 是 使 用 MAP API 来 显示 地 图 ， 然 后 使 用 定位 API 来 获取 设备 的 当前 定位 信息 以 在 
MAP 上 设置 设备 的 当前 位 置 ， 用 户 定位 会 随 着 用 户 的 位 置 移动 而 发 生 改 变 。 
首先 需要 一 个 继承 了 MapActivity 的 Activity 类 ， 例 如 下 面 的 代码 。 


class MyGPSActivity extends MapActivity { 


} 


要 成 功 引用 MAP API， 还 必须 先 在 AndroidManifest.xml 中 定义 如 下 信息 。 
<uses-library android:name="com.google.android.maps" /> 


(2) 使 用 MapView 


要 让 地 图 显示 ， 需 要 将 MapView 加 入 到 应 用 中 来 。 例 如 ， 在 布局 文件 Cmain.xml) 中 加 
入 如 下 代码 。 


<com.google.android.maps.MapView 
android:id="@+id/myGMap" 
android:layout width="fill parent" 
android:layout_ height="fill parent" 
android:enabled="true" 
android:clickable="true" 
android:apiKey="API Key String" 

/> 


另外 ， 要 使 用 MAP 服务 的 话 ， 还 需要 一 个 API key。 可 以 通过 如 下 方式 获取 API key。 

1) 找到 “USER HOME\Local Settings\Application Data\Android ”目录 下 的 “debug. 
keystore” 文 件 。 

2) 使 用 keytool 工具 来 生成 认证 信息 (MD5)， 使 用 如 下 命令 行 。 


keytool -list -alias androiddebugkey -keystore <path to debug keystore>.keystore -storepass 
android -keypass android 


3) 打开 “Sign Up for the Android Maps API” 页 面 ， 输 入 之 前 生成 的 认证 信息 (MD5) 后 
将 获取 到 你 的 API key。 
4) 蔡 换 上 面 AndroidManifest.xml 配置 文件 中 “API Key_String ”为 刚才 获取 的 API key。 


注意 ， 上 面 获取 API key 的 介绍 比较 简单 ， 在 本 章 后 面 的 内 容 中 ， 将 通过 一 个 具体 实例 
的 实现 过 程 来 演示 获取 API key 的 方法 。 


接 下 来 继续 补 全 MyGPSActivity 类 的 代码 。 例 如 下 面 的 代码 。 
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class MyGPSActivity extends MapActivity { 
(QOverride 


public void onCreate(Bundle savedInstanceState) { 
/创建 并 初始 化 地 图 
gMapView = (MapView) fndViewById(R.id.myGMap); 
GeoPoint p = new GeoPoint((int) (lat * 1000000), (inb (long * 1000000)); 
gMapView.setSatellite(true); 
mc = gMapView.getController(); 
mce.setCenter(p); 


Imc.SetZoom(14); 


} 
另外 ， 如 果 要 使 用 定位 信息 的 话 ， 必 须 设 置 一 些 权 限 ， 在 AndroidManifest.xml 中 的 具体 
配置 如 下 。 


<uses-permission android:name="android.permission.INTERNET"></uses-permission> 


<uses-permission 
android:name="android.permission.ACCESS COARSE LOCATION"></uses-permission> 

<uses-permission 
android:name="android.permission.ACCESS FINE LOCATION"></uses-permission> 


(3) 使 用 定位 管理 器 
可 以 通过 使 用 Context.getSystemService0 方 法 ， 传 入 Context.LOCATION_SERVICE 参数 
获取 定位 管理 器 的 实例 。 例 如 下 面 的 代码 。 


LocationManager lm = (LocationManager) getSystemService(Context.LOCATION SERVICE); 


之 后 ， 需 要 将 原来 的 MyGPSActivity 对 象 做 一 些 修 改 ， 让 它 实现 一 个 LocationListener 接 
， 使 其 能 够 监听 定位 信息 的 改变 。 


class MyGPSActivity extends MapActivity implements LocationListener { 


ublic void onLocationChanged(Location location) {} 

public void onProviderDisabled(String provider) {} 

public void onProviderEnabled(String provider) {} 

public void onStatusChanged(String provider, int status, Bundle extras) {} 
protected boolean isRouteDisplayed() { 

return false; 

} 

} 


添加 一 些 代码 ， 对 LocationManager 进行 一 些 初始 化 工作 ， 并 在 它 的 onCreate() 方 法 中 注 
册 定 位 监听 器 。 
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(@Override 

public void onCreate(Bundle savedInstanceState) { 

LocationManager lm = (Location Manager)getSystemService(Context.LOCATION SERVICE); 
lm.requestLocationUpdates(Location Manager.GPS PROVIDER, 1000L, 500.0f, this); 


} 


此 时 代码 中 的 onLocationChanged() 方 法 就 会 在 用 户 的 位 置 发 生 500 米 距 离 的 改变 之 后 进 


行 调 ) 


。 这 里 默认 使 用 的 LocationProvider 是 “gps”(GSP PROVIDER)， 但 是 可 以 根据 你 的 


需要 ， 使 用 特定 的 Criteria 对 象 调用 LocationManger 类 的 getBestProvider(0 方 法 获取 其 他 的 


LocationProvider。 以 下 代码 是 onLocationChanged 方法 的 参考 实现 。 


通过 上 面 的 代码 ， 获 取 了 当前 的 新 位 置 并 更 新 地 


public void onLocationChanged(Location location) { 
if (location != null) { 
double lat = location.getLatitude(); 
double Ing = location.getLongitude(); 
p= new GeoPoint((int) lat * 1000000, (int) Ing * 1000000); 
mc.animateTo(p); 
} 
} 


现 


上 的 位 置 显示 。 还 可 以 为 应 用 程序 添 


加 一 些 诸如 缩放 效果 、 地 图 标注 和 文本 等 功能 。 


(4) 添加 缩放 控件 


将 缩放 控件 添加 到 地 图 上 */ 

ZoomControls zoomControls = (ZoomControls) gMapView.getZoomControls(); 
ZoomControls.SetLayoutParams(new ViewGroup.LayoutParams(LayoutParams.WRAP CONTENT， 
LayoutParams.WRAP CONTENT)); 

gMapView.addView(zoomControls); 

gMapView.displayZoomControls(true); 


(5) 添加 Map Overlay 
来 到 最 后 一 步 ， 添 加 Map Overlay〈 层 )。 通 过 下 面 的 代码 可 以 定义 一 个 Overlay。 


class MyLocationOverlay extends com.google.android.maps.Overlay { 
(QOverride 
public boolean draw(Canvas canvas, MapView mapView, boolean shadow, long when) { 
super.draw(canvas, mapView, shadow); 
Paint paint = new Paint(); 
入 将 经 纬度 转换 成 实际 屏幕 坐标 */ 


Point myScreenCoords = new Point(); 


mapView.getProjection().toPixels(p, myScreenCoords); 

paint.setStroke Width(1); 

paint.setARGB(255, 255, 255, 255); 

paint.setStyle(Paint. Style.STROKE); 

Bitmap bmp = BitmapFactory.decodeResource(getResources(), R.drawable.marker); 
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canvas.drawBitmap(bmp, myScreenCoords.x, myScreenCoords.y, paint); 
canvas.drawText("how are you...", myScreenCoords.x, myScreenCoords.y, paint); 


return true; 


} 
} 


上 面 的 Overlay 会 在 地 图 上 显示 一 个 “Here Iam” 的 文本 ， 然 后 把 这 个 Overlay 添加 到 地 
图 上 去 。 


MyLocationOverlay myLocationOverlay = new MyLocationOverlay(); 
List<Overlay> list = gMapView.getOverlays(); 
list.add(myLocationOverlay); 


3. 使 用 前 的 设置 

Google 地 图 给 人 们 的 生活 带 来 了 极 大 的 方便 。 例 如 可 以 通过 Google 地 图 查找 商户 信息 、 
查看 地 图 和 获取 行车 路 线 等 。Android 平台 也 提供 了 一 个 map 包 (com.google.android.maps)， 
通过 其 中 的 MapView 就 能 够 方便 的 利用 Google 地 图 的 资源 来 进行 编程 。 在 使 用 前 需要 预先 
进行 如 下 必要 的 设置 。 

(1) 添加 maps.jar 到 项 目 

在 Android SDK 中 ， 以 JAR 库 的 形式 提供 了 和 MAP 有 关 的 API， 此 JAR 库 位 与 
“android-sdk-windows\add-ons\google apis-4” 目 录 下 。 要 把 mapsjar 添加 到 项 目 中 ， 可 以 在 项 
目 属 性 中 的 “Android” 栏 中 指定 使 用 包含 Google API 的 Target 作为 项 目的 构建 目标 。 如 
9-7 所 示 。 


; 


Properties for CurrentLoc 


ationWithllap 


type filter text Android CI Or 
Resource 
Project Build Target 
3 im Ve Fn [mn | 
由 .Java Code Style 口 Android 1.1 Android Dpen Source Project | 2 
. ; 口 Android 1.5 Android Dpen Source Project 1 3 
:Jeve Conpiler 口 ndroid 1.6 Android Open Source Project 1.6 4 
由 -Java Bditor 日 aaroia 2.0 Android Open Source Project 2.0 5 
Javadoc Location 口 Mndroid 2.0.1 Android Dpen Source Project 2.0.1 6B 
Project References Google AFIs Google Inc. 1.6 4 
RuryDebug Settings 口 Google AFIs Google Inc. 2.0 5 
由 -Task Repository 口 Google AFIs Google Inc. 2.0.1 6B 
Task Tags 口 Goo8gle HFIs Google Ine. 2.0.1 B 
Validation 口 Google AFIs Google Inc. 2.0.1 6 
WikiText 


Restore Defaults | Apply | 
5 oo | 


图 9-7 在 项 目 中 包含 Google API 
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将 地 图 租 入 到 应 用 
过 使 用 MapActivity 和 MapView 控件 ， 可 以 轻松 地 将 地 图 嵌入 到 应 用 程序 当中 。 在 此 
Ee ， 需 要 将 Google API 添加 到 构建 路 径 中 。 方 法 是 在 图 9-7 所 示 界 面 中 选择 “Java Build 
Path”， 然 后 在 Target 中 勾 选 Google API， 设 置 项 目 中 包含 Google API。 如 图 9-8 所 示 。 


苷 Properties for UserCurrentLocatioriap 


type filter text Android ~ -vw 
:Resource 
Android Project Build Target 
Builders 


Java Build Path 


Amndroid Dpen Source Pro... 


由 Java Code Style 1.1 2 
由 .Java Compiler 口 Android 1.5 Android Dpen Source Pro... 1.5 3 
本 Java Editor 口 Android 1.6 Android Dpen Source Pro... 1.6 4 
: 3 OMndroid 2.0 Mndroid Dpen Source Pro... 2.0 5 
rede Locetion, 口 inadroid 2.0.1 hndroid Open Source Pro... 2.0.1 6 
Project References indroid 2.1... hndroid Open Source Pro... 2.1- 7 

Run/Debug Settings 口 cooaas APIs Google Ine. 2.1-... 7 
由 Task Repository 口上 Rid 2. 2 Android Dpen Source Fro... 2.2 8 
. Task Tags Goog 2.2 8 
"Validation 

WikiText 


Mndroid + Google AFIs 


-Library 
厂 Is Library 


oo 
© cca | 


图 9-8 将 Google API 添加 到 构建 路 径 


(3) 获取 Map API 密 铀 

在 利用 MapView 之 前 ， 必 须要 先 申 请 一 个 Android Map APIKey。 具 体 步 又 如 下 。 
第 1 步 : 找到 你 的 debug.keystore 文件 ， 通 常 位 于 如 下 目录 。 

C:\Documents and Settings\ 你 的 当前 用 户 \Local Settings\Application Data\Android 
第 2 步 : 获取 MD5 指纹 。 运 行 cmd.exe， 执 行 如 下 命令 获取 MD5 指纹 。 


>keytool -list -alias androiddebugkey -keystore "debug.keystore 的 路 径 " -storepass android -keypass android 


例如 ， 作 者 机 器 输入 如 下 命令 。 


keytool -list -alias androiddebugkey -keystore "C:\Documents and Settings\Administrator\. android\ 
debug.keystore" -storepass android -keypass android 


此 时 系统 会 提示 输入 keystore 密码 ， 这 时 候 输 入 android， 系 统 就 会 输出 我 们 申请 到 的 
MD5 认证 指纹 。 如 图 9-9 所 示 。 
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注意 : 因为 在 CMD 中 不 能 直接 复制 、 粘 贴 使 用 CMD 命令 ， 这 样 很 影响 编程 效率 ， 所 以 
笔者 使 用 了 第 三 方 软件 PowerCmd 来 代替 机 器 中 自 带 的 CMD 工具 。 


Cmd - PowerCmd 


File View 十 二 Edit Help 
| EDA 
search| 司 z| 


图 9-9 获取 的 认证 指纹 


第 3 步 : 申请 Android map 的 API Key。, 打 开 浏 览 器 ,输入 下 面 的 网 址 :“http://code.google. 
com/intl/zh-CN/android/maps-api-signup. html”， 如 图 9-10 所 示 。 


If you use different keys for signing development builds and release builds, You will need to obtain a separate Maps APl key for each certificate. Each key will on 
the corresponding certificate. 


You also need a Google Account to get a Maps API key, and Your API key will be connected to your Google Account 


IAndroid Maps APIs Terms of Service 


Last Updated: October 13, 2008 


IThanks for your interest in the Android Maps APIs. The Android Japs 
APIs are a collection of services (including, but not limited to, the 
“com. google. android. maps. NapYiew” and “android. location. Geocoder” 
classes) that allow you to include maps, geocoding, and other content 
from Google and its content providers in your Android applications. 
IThe Android Maps APIs explicitly do not include any driving directions 
data or local search data that may be owmed or licensed by Google. 


1. Your relationship with Google. 
1.1. Your use of any of the Android Maps APIs (referred to in this 
document as the “llaps API(s)” or the “Service”) is subject to the 到 


La | have read and agree with the terms and conditions (printable version) 


My certificate's MD5 fingerprint: |58: AE: 06:C3:5F:4F:FF:D6:89:C5:CD:76:04:D0:2F:AF 


Generate API Key | 


wal 
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在 Google 的 Android Map API Key 申请 页 面 上 输入 图 9-10 中 得 到 的 MD5S 认证 指纹 ， 按 下 
“Generate APIKey” 按 钮 后 即 可 转 到 下 面 的 这 个 画面 , 得 到 我 们 申请 到 的 API Key。 如 图 9-11 所 示 。 


Gooslke So 


mca 八 隐 三 页 > Coocle 地 四 4F1 > Geog 志 国 AFI 注 而 
十 谢 晓 注册 Androld 好 末 AP1 刻 钢 : 
息 的 时 外 是 ， 
or?7tt yp OM LTC ye FO 
贡 转 悄 脖 用 竹 所 有 后 用 局 下 扒 休 所? 语 庆 书 进行 难 这 和 占用 岂 序 、 
TT 
笠 国 县 一 个 wm 效 交 的 对 济 ， 业 煌 名 了 媚 地 辣 D 抱 


Com google andrcad naps EapTier 
L411 parent 


"4 parenr 
"OT NGION LIXeNCMTUh SCN rt FO" 


wal 
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至 此 ， 成 功 获取 了 一 个 API Key， 编 程 前 的 整个 准备 工作 也 完成 了 。 
4. 使 用 Map API 密 钥 的 基本 流程 
通过 上 面 的 讲解 ,我 们 已 经 申请 到 了 一 个 Android Map APIKey， 下 面 
API 密 钥 实现 编程 的 基本 流程 。 
第 1 步 : 在 文件 AndroidManifest.xml 中 声明 权限 。 在 Anroid 系统 中 ， 如 果 程 序 执行 需要 
读 取 到 安全 敏感 的 项 目 ， 那 么 必须 在 AndroidManifestxml 中 声明 相关 权限 请 求 ， 比 如 这 个 地 
图 程序 需要 从 网 络 读 取 相关 数据 。 所 以 必须 声明 android.permission.INTERNET 权限 。 具 体 方 
法 是 在 文件 AndroidManifest.xml 中 添加 如 下 代码 。 


开始 讲解 使 用 Map 


<uses-permission android:name="android.permission.INTERNET" /> 


另外 ， 因 为 maps 类 不 是 Android 启动 的 缺 省 类 ， 所 以 还 需要 在 文件 AndroidManifest.xml 
的 application 标签 中 声明 要 用 maps 类 。 


<uses-library android:name="com.google.android.maps" /> 


下 面 是 基本 的 AndroidManifest.xml 文件 代码 。 


<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
<application android:icon="(@drawable/icon" android:label="(@string/app_name"> 
<uses-library android:name="com.google.android.maps" /> 
</application> 
<uses-permission android:name="android.permission.INTERNET" /> 


</manifest> 


第 2 步 : 在 main.xml 主 文件 中 完成 Layout。 下 面 开始 着 手 来 完成 界面 ， 假 设 设置 要 显示 
杭州 的 卫星 地 图 ， 并 在 地 图 上 方 有 5 个 按钮 ， 分 别 可 以 放大 地 图 、 缩 小 地 图 或 者 切换 显示 模 
式 〈 卫 星 、 交 通 和 街景 )。 即 整个 界面 主要 由 2 个 部 分 组 成 ， 上 面 是 一 排 5 个 按钮 ， 下 面 是 
Map View。 

在 Android 中 ，LinearLayout 是 可 以 互相 嵌 套 的 ， 在 此 可 以 把 上 面 $ 个 按钮 放 在 一 个 子 
LinearLayout 里 边 ( 子 LinearLayout 的 指定 可 以 由 android:addStatesFromChildren="true" 实 现 )， 
然后 再 把 这 个 子 LinearLayout 加 到 外 面 的 父 LinearLayout 里 边 。 
第 3 步 : 完成 主 Java 程序 代码 。 


9.4.2 具体 实现 


本 实例 的 主 文件 是 example4.java 和 main.xml， 下 面 分 别 介绍 其 实现 流程 。 

1. 文件 example4.java 

在 文件 example4.java 中 , 通过 EditText 来 输入 坐标 的 经 度 和 维度 , 将 坐标 转换 为 GeoPoint 
对 象 后 ， 再 利用 MapController 对 象 的 animateTo0) 方 法 将 地 图 的 中 心 点 移 到 GeoPoint 坐标 上 。 
文件 example4.java 的 具体 实现 代码 如 下 。 


public class example4 extends MapActivity 


{ 
private MapController mMapController01; 
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private MapView mMapView0!1; 
private Button mButton01,mButton02,mButton03; 
private EditText mEditTextO1; 
private EditText mEditText02; 
private int intZoomLevel=0; 
久 Map 启动 时 的 默认 坐标 沁 
private double dLat=120.391177; 
private double dLng=39.9067452; 


(QOverride 
protected void onCreate(Bundle icicle) 


{ 
super.onCreate(icicle); 
setContentView(R.layout.main); 


/# 创建 MapView 对 象 */ 

mMapView01 = (MapView)findViewByld(R.id.myMapView!); 
mMapController01 = mMapView01.getController(); 
/* 设置 MapView 的 显示 选项 (卫星 、 街 道 ) */ 
mMapView01.setSatellite(false); 
mMapView01.setStreetView(true); 

雍 默认 放大 的 层级 * 

intZoomLevel = 17; 
mMapController01.setZoom(intZoomLevel); 

此 设置 Map 的 中 点 为 默认 经 纬度 */ 

refresh MapView(); 


mbEditText01 = (EditText)findViewById(R.id.myEditl); 
mbEditText02 = (EditText)findViewById(R.id.myEdit2); 


/# 送出 查询 的 Button */ 
mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton01.setOnClickListener(new Button.OnClickListener() 


{ 


(QOverride 
public void onClick(View v) 
{ 
放 经 纬度 空白 检查 */ 
if(mEditText01.getText().toString().equals("") 
mEditText02.getText().toString().equals("")) 
{ 
showDialog(" 经 度 或 纬度 填写 不 正确 !"); 
} 
else 
{ 
上 # 取得 输入 的 经 纬度 */ 
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dLng=Double.parseDouble(mEditText01.getText().toString()); 
dLat=Double.parseDouble(mEditText02.getText().toString()); 
必 根据 输入 的 经 纬度 重 置 地 图 */ 

refresh MapView(); 


wh 


} 
有 


记 放大 地 图 的 按钮 */ 
mButton02 = (Button)findViewById(R.id.myButton2); 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 
Override 
public void onClick(View v) 
{ 
intZoomLevel++; 
if(intZoomLevel>mMapView01.getMaxZoomLevel()) 
{ 
intZoomLevel = mMapView01.getMaxZoomLevel(); 
} 
mMapController01.setZoom(intZoomLevel); 
} 
及 


洗 缩小 地 图 的 按钮 */ 
mButton03 = (Button)findViewById(R.id.myButton3); 
mButton03.setOnClickListener(new Button.OnClickListener() 
{ 

(QOverride 

public void onClick(View v) 


{ 
intZoomLevel--; 
if(intZoomLevel<1) 
{ 
intZoomLevel = 1; 


} 
mMapController01.setZoom(intZoomLeve)l); 


/* 时 


wh 


地 图 的 方法 */ 
public void refresh MapView() 


{ 
GeoPoint p= new GeoPoint((int)(dLat* 1E6), (int)(dLng* 1E6)); 
mMapView01.displayZoomControls(true); 
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/# 将 Map 的 中 点 移 人 至 GeoPoint */ 
mMapController01.animateTo(p); 
mMapController01.setZoom(intZoomLevel); 


(@Override 
protected boolean isRouteDisplayed() 


{ 


return false; 


/* 显示 Dialog 的 方法 */ 

private void showDialog(String mess){ 
new AlertDialog.Builder(example4.this).setTitle("Message") 
.SetMessage(mess) 


=31] 


.SetNegativeButton(" 确 定 


{ 
public void onClick(DialogInterface dialog, int which) 


, new DialogInterface.OnClickListener() 


2. 文件 main.xml 
文件 main.xml 是 一 个 布局 文件 ， 主 要 代码 如 下 。 


</Button> 

<!-- Google MapView Widget --> 

<com.google.android.maps.MapView 
android:id="(@+tid/myMapView1" 
android:layout width="fill parent" 
android:layout height="fill parent" 
android:layout x="O0px" 
android:layout_ y="102px" 
android:enabled="true" 
android:clickable="true" 
android:apiKey="0by7ffx8]jX0A LWXeKCMTWAh8CqHAlqvzetFqjQ" 

> 

</com.google.android.maps. MapView> 

</AbsoluteLayout> 


上 述 代码 的 核心 是 android:apiKey="0by7ffx8jX0A _LWXeKCMTWAh38CqHAlqvzetFqjQ" 
即 调用 了 Android Maps API key。 
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Google 提供 了 一 个 Geocoder 服务 , 在 非 商 / 


对 象 。 在 本 节 的 内 容 中 ， 将 通过 


例 的 具体 实现 流程 。 


9.5.1 Geocoder 基 础 


情况 下 ，Geocoder 能 够 反 查 Address (地址 ) 


个 具体 实例 的 实现 ， 介 绍 在 Android 中 通过 Geocoder 实现 
地 址 反 查 功 能 的 流程 。 本 实例 的 源 代 码 保存 在 “光盘 :\daima\9\example5”， 下面 开始 讲解 本 实 


什么 是 Geocoder， 简 单 说 就 是 根据 某 个 key〈 键 值 ) 寻找 地 到 


1H 


! 位 置 的 坐标 ， 这 个 key 可 


以 是 地 址 。 近 年 来 Google Map API 做 了 很 多 改进 ， 已 经 自动 整合 了 中 文 地 址 的 定位 功能 ， 


不 再 需要 以 前 的 mashup 了 。 以 前 只 有 6 个 
望 。 虽然 存在 每 日 只 能 50000 次 查询 的 限制 , 但 是 一 


cache 改善 用 户 体 验 。 


国家 提供 了 街道 级 别 的 定位 ，Google 从 不 让 人 失 
股 也 足够 了 , 何况 还 有 客户 端的 built-in 


这 个 key 还 可 以 是 ip 地址 , 例如 做 一 个 library 可 以 读 取 本 地 数据 库 转 换 ip 为 坐标 , 这 个 
坐标 只 是 城市 中 心 的 坐标 ， 不 过 一 般 也 够 用 了 。 同 时 library 也 将 提供 基于 http 请 求 的 公共 服 
务 ， 地 址 定位 和 ip 定位 现在 都 很 容易 实现 了 。 


9.5.2 具体 实现 


本 实例 的 主 文件 是 example5.java， 功 能 是 在 地 图 中 定位 输入 的 地 址 。 首 先 通过 方法 
getGeoByAddress() 获取 输入 地 址 的 字符 串 ， 然 后 传 入 这 个 地 址 ， 并 


.使 用 


Geocoder.getFromLocationName() 方 法 获取 从 Google 服务 器 找到 的 结果 。 文 件 example5.java 


的 具体 实现 代码 如 下 。 


public class exampleS extends MapActivity 


{ 


private MapController mMapController01; 
private MapView mMapView0!1; 


private Button mButton01,mButton02,mButton03; 
private EditText mEditTextO1; 

private int intZoomLevel=15; 

private String TAG = "HIPPO GEO DEBUG"; 


(QOverride 


protected void onCreate(Bundle icicle) 


{ 


super.onCreate(icicle); 


setContentView(R.layout.main); 


mbEditText01 = (EditText)findViewById(R.id.myEditTextl); 


mEditText01.setText 
( 
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getResources().getText(R.string.str_ default address).toString() 
); 


/# 创建 MapView 对 象 */ 
mMapView01 = (MapView)findViewByld(R.id.myMapView!); 
mMapController01 = mMapView01.getController(); 


/# 设置 MapView 的 显示 选项 (卫星 、 街 道 )*/ 
mMapView01.setSatellite(true); 


mMapView01.setStreetView(true); 


mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
if(mEditText01.getText().toString()!="") 
{ 
refresh MapViewByGeoPoint 
( 
getGeoByAddress 
( 
mEditText01.getText().toString() 
),mMapView01,intZoomLevel,true 
); 


} 
J 


Vr st 
mButton02 = (Button)findViewById(R.id.myButton2); 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View v) 
{ 
intZoomLevel++; 
if(intZoomLevel~>mMapView01.getMaxZoomLevel()) 
{ 
intZoomLevel = mMapView01.getMaxZoomLevel(); 


} 
mMapController01.setZoom(intZoomLevel); 


2 
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上 # 缩 小 忆 
mButton03 = (Button)findViewById(R.id.myButton3); 
mButton03.setOnClickListener(new Button.OnClickListener() 
{ 

Override 

public void onClick(View v) 

{ 


// TODO Auto-generated method stub 
intZoomLevel--; 
if(intZoomLevel<1) 
{ 
intZoomLevel = 1; 
} 
mMapController01.setZoom(intZoomLevel); 
} 
); 


/初次 查询 地 点 */ 
refresh MapViewByGeoPoint 
( 
getGeoByAddress 
( 
getResources().getText(R.string.str default address).toString() 
),mMapView01,intZoomLevel,true 


, 
} 
private GeoPoint getGeoByAddress(String strSearchAddress) 
{ 
GeoPoint gp = null; 
ty 
if(strSearchAddress!="") 
{ 


Geocoder mGeocoder01 = new Geocoder(example5 .this, Locale.getDefault()); 
List<Address> lstAddress = mGeocoder01.getFromLocationName(strSearchAddress, 1); 
if (llstAddress.isEmpty()) 
{ 
// Address[addressLines=[0:"U.S PIZZA",l:"1lSth Main Rd, Phase I, J P 
Nagar",2:"Bengaluru, Karnataka",3:"India"l,feature=U.S PIZZA,admin=Karnataka,sub-admin=Bengaluru,locality= 
Bengaluru,thoroughfare=15th Main Rd,postalCode=null,countryCode=IN,countryName= India,hasLatitude= true, 
latitude=18.508933,hasLongitude=true,longitude=73.8042,phone=null,url=null,extras=null] 
/* 
for (int i= 0; i< 1stAddress.size(); ++1) 
{ 
Address adsLocation = lstAddress.get(i); 
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Log.i(TAG, "Address found = " + adsLocation.toString()); 
lat = adsLocation.getLatitude(); 
lon = adsLocation.getLongitude(); 

} 

S$/ 

Address adsLocation = lstAddress.get(0); 

double geoLatitude = adsLocation.getLatitude()*1E6; 


double geoLongitude = adsLocation.getLongitude()*1E6; 
gp =new GeoPoint((int) geoLatitude, (int) geoLongitude); 
} 


else 


{ 
Log.i(TAG, "Address GeoPoint NOT Found."); 


} 
catch (Exception e) 
{ 
e.printStack Trace(); 
} 
return gp; 
} 


public static void refresh MapViewByGeoPoint(GeoPoint gp, MapView myv, int zoomLevel, boolean 
blfSatellite) 


try 

{ 
myv.displayZoomControls(true); 
/# 取得 MapView 的 MapController */ 
MapController mc = myv.getController(); 
从 移 至 该 地 理 坐 标 地 址 */ 


mc.animateTo(gp); 


席 放大 地 图 层级 * 


mc.SetZoom(zoomLevel); 


/* 设置 MapView 的 显示 选项 〈 卫 星 、 街 道 ) */ 
if(bIfSatellite) 
{ 


myv.setSatellite(true); 


myv.setStreetView(true); 


} 


else 


{ 


myv.setSatellite(false); 
480O 看 看 


} 


} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 
} 


(QOverride 
protected boolean isRouteDisplayed() 


{ 
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// TODO Auto-generated method stub 
return false; 


} 
} 


执行 后 能 够 实现 地 址 反 查 处 理 ， 如 图 9-12 所 示 。 


呜 阐 个 9:58 ww 
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图 9-12 ”执行 效果 


9.6 Diredions Route DODODODOD 


在 Android SDK 中 ,可 以 调用 手机 内 置地 图 程序 来 传递 导航 坐标 来 规划 路 径 。 在 本 节 的 内 容 


中 ， 将 通过 一 个 


具体 实例 的 实现 ， 


介绍 在 Android 中 通过 Directions Route 实现 路 径 导 航 的 流程 。 


本 实例 的 源 代码 保存 在 “光盘 :\daima\9\example6”， 下 面 开始 讲解 本 实例 的 具体 实现 流程 。 


9.6.1 ”实现 原理 


在 具体 实现 上 ， 先 调用 getLocationProvider() 方 法 来 获取 当前 Location( 位 置 )， 以 取得 当 
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前 所 在 位 置 的 地 理 坐 标 。 并 通过 提供 的 EditText 组 件 来 让 用 户 输入 地 址 ， 通 过 地 址 取得 目的 
地 的 地 理 坐 标 。 通 过 2 个 GeoPoint 对 象 ， 并 通过 Intent 方式 来 调用 内 置 的 地 图 程序 。 在 具体 
实现 上 ， 要 把 握 如 下 2 点 。 
口 以 Url.parse0 方 法 传 入 Google Map 的 路 径 规划 参数 方法 。 
口 根据 于 机 的 移动 状态 ， 更 改 并 更 新 当前 GeoPoint 的 方法 。 

只 要 设置 好 了 起 点 GeoPoint 和 终点 GeoPoint， 通 过 重组 Google Map 的 GET 传输 参数 ， 
便 可 以 传 入 Google Map 显示 。 实 际 上 ，map.google.com 能 够 接受 的 经 纬度 需要 以 “精度 ， 维 
度 ” 的 字符 串 格 式 来 传递 ， 所 以 编写 了 一 个 GeoPointToString0) 来 重组 GeoPoint 的 经 纬度 信 


9.6.2 具体 实现 
本 实例 的 主 文件 是 example6.java， 下 面 开始 讲解 其 具体 实现 流程 。 


1) 创建 MapView 对 象 ， 并 创建 LocationManager 对 象 取得 系统 定位 服务 ， 然 后 设置 
MapView 对 象 的 显示 选项 (卫星 、 街 道 )。 具 体 实现 代码 如 下 


@Override 

protected void onCreate(Bundle icicle) 

{ 
// TODO Auto-generated method stub 
super.onCreate(icicle); 
setContentView(R.layout.main); 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 


mbEditText01 = (EditText)findViewById(R.id.myEditTextl); 
mEditTextO1.setText 
( 
getResources().getText 
(R.string.str_default address).toString() 
); 


/# 创建 MapView 对 象 */ 
mMapView01 = (MapView)findViewByld(R.id.myMapView!); 
mMapController01 =mMapView01.getController(); 


/* 设置 MapView 的 显示 选项 (卫星 、 街 道 ) */ 口 
mMapView01.setSatellite(true); 
mMapView01.setStreetView(true); 


谨 放大 的 层级 * 
intZoomLevel = 15; 
mMapController01.setZoom(intZoomLevel); 


/# 创建 LocationManager 对 象 取得 系统 LOCATION 服务 */ 
mLocationManager01 = 
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(Location Manager)getSystemService(Context.LOCATION SERVICE); 


* 自 定义 方法 ， 访 问 Location Provider， 
* 并 将 之 存储 在 strLocationProvider 当中 
革 

getLocationProvider(); 


/* 传 入 Location 对 象 ， 显 示 于 MapView */ 
fromGeoPoint = getGeoByLocation(mLocation01); 
refreshMapViewByGeoPoint(fromGeoPoint, 
mMapView01, intZoomLevel); 


/* 创建 LocationManager 对 象 ， 监 听 

* Location 更 改 时 事件 ， 更 新 MapView*/ 
mLocation Manager01.requestLocationUpdates 
(strLocationProvider, 2000, 10, mLocationListener01); 


2) 定义 toGeoPoint 对 象 获取 用 户 要 前 往 地 址 的 GeoPoint 对 象 ， 传 入 路 径 规 划 所 需要 的 
地 标 地 址 。 有 具体 实现 代码 如 下 。 


mButton01 = (Button)findViewById(R.id.myButton!); 
mButton01.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
public void onClick(View V) 
{ 
if(mEditText01.getText().toString()!="") 
{ 
上 # 取得 用 户 要 前 往 地 址 的 GeoPoint 对 象 ** 
toGeoPoint = 


getGeoByAddress(mEditText01.getText().toString()); 


雍 路 径 规 划 Intent */ 
Intent intent = new Intent(); 
intent.setAction(android.content.Intent.ACTION VIEW); 


雍 传 入 路 径 规 划 所 需要 的 地 标 地 址 党 

intent.setData 

( 
Uri.parse("http://maps.google.com/maps? 人 = d&saddr="+ 
GeoPointToString(fromGeoPoint)+ 
"&daddr="+GeoPointToString(toGeoPoint)+ 
"&hl=cn" 十 
") 

); 

startActivity(intent); 
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} 
) 
)); 


3) 定义 mButton02 按钮 处 理事 件 ， 实 现 地 图 放大 处 理 。 定 义 mButton03 按钮 处 理事 件 ， 
实现 地 图 缩小 处 理 。 具 体 代 码 如 下 。 


/# 放大 地 图 */ 
mButton02 = (Button)findViewById(R.id.myButton2); 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 
Override 
public void onClick(View V) 
{ 
intZoomLevel++; 
if(intZoomLevel>mMapView01.getMaxZoomLevel()) 
{ 
intZoomLevel = mMapView01.getMaxZoomLevel(); 


} 
mMapController01.setZoom(intZoomLevel); 
)); 


UU 
有 


庆 缩小 地 图 所 
mButton03 = (Button)findViewById(R.id.myButton3); 
mButton03.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View V) 
{ 
intZoomLevel--; 
if(intZoomLevel<1) 
{ 
intZoomLevel = 1; 


} 
mMapController01.setZoom(intZoomLevel); 


»); 
} 
4) 捕捉 当 手 机 GPS 坐标 更 新 时 的 
getMyLocation 对 象 。 有 具体 代码 如 下 。 
庆 捕捉 当 手 机 GPS 坐标 更 新 时 的 事件 光 


public final LocationListener mLocationListener01 = 


有 件 ， 当 手机 收 到 位 置 更 改 时 ， 将 定位 位 置 传 入 


山中 


new LocationListener() 


{ 
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Override 
public void onLocationChanged(Location location) 


{ 


举 当 手 机 收 到 位 置 更 改 时 ， 将 location 传 入 getMyLocation */ 


mLocation01 = location; 


fromGeoPoint = getGeoByLocation(location); 
refresh MapViewByGeoPoint(fromGeoPoint, 
mMapView01, intZoomLevel); 


(QOverride 
public void onProviderDisabled(String provider) 


{ 


mLocation01 = null; 


Override 
public void onProviderEnabled(String provider) 


{ 


(QOverride 
public void onStatusChanged(String provider, 
int status, Bundle extras) 


) 
翅 


| 


NN 
4 


5) 定义 方法 getGeoByLocation(Location location)， 当 传 入 定位 位 置 对 象 时 取 回 其 
GeoPoint 对 象 。 具 体 代码 如 下 。 


族 传 入 Location 对 象 ， 取 回 其 GeoPoint 对 象 */ 
private GeoPoint getGeoByLocation(Location location) 
{ 
GeoPoint gp = null; 
try 
{ 
/* 当 Location 存在 */ 
if (location != null) 


{ 


double geoLatitude = location.getLatitude()*1E6; 
double geoLongitude = location.getLongitude()*1E6; 
gp =new GeoPoint((int) geoLatitude, (int) geoLongitude); 
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} 
} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 


return gp; 


} 


6) 定义 方法 getGeoByAddress(String strSearchAddress)， 当 输入 地 址 时 取得 其 GeoPoint 
对 象 。 具体 代码 如 下 。 


庆 输入 地 址 ， 取 得 其 GeoPoint 对 象 六 
private GeoPoint getGeoByAddress(String strSearchAddress) 
{ 
GeoPoint gp = null; 
try 
{ 
if(strSearchAddress!="") 


{ 


Geocoder mGeocoder01 = new Geocoder 


(example6.this, Locale.getDefault()); 


List<Address> lstAddress = mGeocoder01.getFromLocationName 
(strSearchAddress, 1); 
if (llstAddress.isEmpty()) 
{ 
Address adsLocation = lstAddress.get(0); 
double geoLatitude = adsLocation.getLatitude()*1E6; 
double geoLongitude = adsLocation.getLongitude()*1E6; 
gp = new GeoPoint((int) geoLatitude, (int) geoLongitude); 
} 
} 
} 
catch (Exception e) 
{ 
e.printStackTrace(); 
} 
return gp; 


} 


7) 定义 方法 refreshMapViewByGeoPoint0 〇 和 和 方法 refreshMapViewByCode(), 传 入 geoPoint 对 象 
更 新 MapView 里 的 Google Map 和 传 入 经 纬度 更 新 MapView 里 的 Google Map。 具 体 代 码 如 下 。 


愉 传 入 geoPoint 更 新 MapView 里 的 Google Map */ 
public static void refresh MapViewByGeoPoint 


486 国 国 


(GeoPoint gp, MapView mapview, int zoomLevel) 


{ 


try 


{ 


} 


mapview.displayZoomControls(true); 
MapController myMC = mapview.getController(); 


myMC.animateTo(gp); 
myMC.setZoom(zoomLevel); 
mapview.setSatellite(false); 


catch(Exception e) 


{ 


e.printStackTrace(); 


访 传 入 经 纬度 更 新 MapView 里 的 Google Map*/ 
public static void refresh MapViewByCode 
(double latitude, double longitude, 


} 


8) 定义 方法 GeoPointToString(GeoPoint gp)， 将 GeoPoint 


MapView mapview, int zoomLevel) 


try 


{ 


} 


GeoPoint p= new GeoPoint((int) latitude, (int) longitude); 
mapview.displayZoomControls(true); 

MapController myMC = mapview.getController(); 
myMC.animateTo(p); 

myMC.setZoom(zoomLevel); 

mapview.setSatellite(false); 


catch(Exception e) 


{ 


e.printStackTrace(); 


形式 返回 。 上 其 体 代码 如 下 。 


/# 将 GeoPoint 里 的 经 纬度 以 String，String 返回 */ 
private String GeoPointToString(GeoPoint gp) 


{ 


String strReturn=""; 
try 


{ 


谍 当 Location 存在 */ 


! 的 经 纬度 以 “String,String” 


加 国 487 


Android 开发 完全 实战 宝典 
if (gp != null) 
{ 
double geoLatitude = (int)gp.getLatitudeE6()/1E6; 
double geoLongitude = (int)gp.getLongitudeE6()/1E6; 
strReturn = String.valueOf(geoLatitude)+","+ 
String.valueOf(geoLongitude); 


} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 


return strReturn; 


} 


9) 定义 方法 getLocationProvider()， 用 于 获取 定位 者 。 具 体 代 码 如 下 。 


/* 取得 LocationProvider */ 
public void getLocationProvider() 
{ 
try 
{ 
Criteria mCriteria01 = new Criteria(); 
mCriteria01.setAccuracy(Criteria. ACCURACY FINE); 
ImCriteria01.setAltitudeRequired(false); 
mCriteria01.setBearingRequired(false); 
mCriteria01.setCostAllowed(true); 
mCriteria01.setPowerRequirement(Criteria. POWER LOW); 
strLocationProvider = 
mLocationManager01.getBestProvider(mCriteria01, true); 


mLocation01 = mLocation Manager01.getLastKnownLocation 
(strLocationProvider); 


} 


catch(Exception e) 


{ 


mTextView01.setText(e.toString()); 
e.printStackTrace(); 


} 
} 


(QOverride 
protected boolean isRouteDisplayed() 


{ 


return false; 
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执行 后 的 效果 如 图 9-13 所 示 ; 单 击 “ 开 始 规划 路 径 ” 按 钮 后 弹出 选择 对 话 框 ， 如 图 9-14 
所 示 ; 在 此 选择 “Maps” 后 弹出 规划 界面 ， 如 图 9-15 所 示 ; 在 第 一 个 文本 框 中 设置 出 发 位 置 ， 
例如 “beijing” 在 第 二 个 文本 框 中 设置 目的 地 位 置 ， 例 如 “tianjin”， 选择 汽车 前 往 标志 按钮 
园 上 |， 如 图 9-16 所 示 ; 单 击 按钮 “Go”， 系统 将 实现 由 北京 出 发 ， 目 的 地 到 天 津 的 线路 规划 ， 
产生 线路 规划 图 ， 最 终 的 执行 界面 如 图 9-17 所 示 。 
GE 


example6 


人 Complete action using 
人 Browser 


pa Maps 
Use by default for this action. 


图 9-13 ”执行 效果 图 9-14 ”选择 对 话 框 


39.904667,116.408198 


图 9-15 规划 界 国 


wal 


9-16 设置 出 发 地 和 目的 地 
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动 面 人 丘 2:02PM 
9 Beijing, China 
Tianjin, China 
网 136km=about1h50m 


© Show on map 


Directions 


全 Head north on 正义 路 /正义 路 
toward 东 长 安 街 / 东 长 安 街 


@ Take the 1st right onto 东 长 安 街 / 
东 长 安 街 a 
ed CC I 


@ Continue straight through 东单 / Pe ee fe | Pe PO Fr 
东单 onto 建国 门 内 大 街 / 建 国门 L L 
内 大 街 ee ep (Pe re | Pe 
@ Continue onto 建国 门 外 大 街 / 建 = 
国门 外 大 街 Fr pad td Pr 


图 9-17 生成 的 线路 规划 
当 单 击 图 9-17 中 的 “Show on map” 后 ， 将 会 在 地 图 中 显示 行走 线路 ， 如 图 9-18 所 示 。 


Zhong 3 
| Head north on 正义 路 /了 
| 安 街 / 东 长 安 竺 


Bl ll | 
ee ee ei ee ee 
pe pe em pe ee ee ee 
nennlnlenlenlm mld 
Ee 


ALT 


图 9-18 地 图 中 的 线路 规划 


注意 : 在 此 事 发 地 和 目的 地 不 能 属于 两 个 不 同 的 国家 ， 否 则 将 会 产生 错误 提示 。 


9.7 LocationListener] MapView|| DODO 


在 现实 应 用 中 ，GPS 的 使 用 越 来 越 广泛 。 但 是 每 一 个 位 置 和 路 况 都 不 是 固定 不 变 的 ， 这 
就 要 求 系统 能 够 根据 各 种 变化 而 变化 ， 不 能 误导 了 人 们 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 
实例 的 实现 ， 介 绍 在 Android 中 实现 GPS 实时 更 新 的 处 理 流 程 。 本 实例 的 源 代 码 保 在 在 “ 光 
盘 :daimaN9\example7”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 
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9.7.1 ”实现 原理 
在 Android SDK 中 , 支持 手机 GPS 定位 事件 处 理 。 在 Android 手机 中 内 置 了 Google Map， 


但 是 不 能 随 着 手机 的 移动 而 更 新 。 在 本 实例 中 ,将 插入 一 个 TextView 和 MapView， 当 手机 移 
动 时 ， 会 触发 内 置 的 GPS 定位 坐标 的 改变 事件 。 只 要 程序 发 现 地 理 位 置 的 变化 ， 便 实时 更 新 
MapView 里 的 Google Map， 并 反 查 地 理 坐 标 系统 的 信息 。 


i 


9.7.2 具体 实现 

本 实例 的 主 文件 是 example7.java， 下 面 开始 介绍 其 实现 流程 。 

1) 先 创建 MapView 对 象 mMapView01， 然 后 创建 LocationManager 对 象 mLocation- 
Manager01 来 获取 系统 LOCATION 服务 。 具 体 代码 如 下 。 


protected void onCreate(Bundle icicle) 
{ 
super.onCreate(icicle); 


setContentView(R.layout.main); 


mTextView01 = (TextView)findViewByld(R.id.myTextView!); 
/# 创建 MapView 对 象 */ 
mMapView01 = (MapView)jfindViewById(R.id.myMapView1l); 


/# 创建 LocationManager 对 象 取得 系统 LOCATION 服务 */ 
mLocationManager01 = 
(Location Manager)getSystemService(Context.LOCATION SERVICE); 


2) 通过 getLocationProvider 取得 当前 Location, 然后 创建 LocationManager 对 象 用 于 监听 
Location 更 改 事件 ， 并 更 新 MapView。 具 体 代 码 如 下 。 


/# 第 一 次 运行 从 Location Provider 取得 Location */ 
mLocation01 = getLocationProvider(mLocation Manager01); 


if(mLocation01!=null) 
{ 
processLocationUpdated(mLocation01); 
} 
else 
{ 
mTextView01.setText 
( 
getResources().getText(R.string.str_err location).toString() 
); 
} 
/* 创建 LocationManager 对 象 ， 监 听 Location 更 改 事件 ， 更 新 MapView */ 
mLocation Manager01.requestLocationUpdates 
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4) 定义 方法 getAddressbyGeoPoint(GeoPoint gp)， 用 
Geocoder 对 象 并 取 昌 
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(strLocationProvider, 2000, 10, mLocationListener01); 


} 


L 体 代码 如 下 。 


public final LocationListener 
mLocationListener01 = new LocationListener() 


{ 
(QOverride 


public void onLocationChanged(Location location) 


{ 
// TODO Auto-generated method stub 


席 当 手 机 收 到 位 置 更 改 时 ， 将 位 置 传 入 取得 地 理 坐标 */ 


processLocationUpdated(location); 


(QOverride 
public void onProviderDisabled(String provider) 
{ 

/#* 当 Provider 已 离开 服务 范 上 


奎屯 


ey 


(QOverride 

public void onProviderEnabled(String provider) 
{ 

} 


(QOverride 
public void onStatusChanged 
(String provider, int status, Bundle extras) 


{ 


} 
了 


放 在 StringBuilder 对 象 中 输出 。 有 具体 代码 如 下 。 
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public String getAddressbyGeoPoint(GeoPoint gp) 
{ 


String strReturn = ""; 
try 


于 获取 定位 地 址 的 信息 。 先 创建 
8 地理 坐标 经 纬度 ， 然 后 判断 地 址 是 否 为 多 行 


通过 LocationListener0 监 听 定 位 信息 ， 当 手机 收 到 位 置 更 改 时 ， 将 位 置 传 入 取得 地 理 


， 最 后 将 获取 的 地 址 组 合 后 
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{ 
谍 当 GeoPoint 不 等 于 空 时 */ 
if (gp != null) 
{ 
诺 创建 Geocoder 对 象 */ 
Geocoder gc =new Geocoder 


(example7 .this, Locale.getDefault()); 


上 取出 地 理 坐 标 经 纬度 */ 
double geoLatitude = (int)gp.getLatitudeE6()/1E6; 
double geoLongitude = (int)gp.getLongitudeE6()/1E6; 


上/# 从 经 纬度 取得 地 址 〈 可 能 有 多 行 地 址 ) 六 
List<Address> lstAddress = 
gc.getFromLocation(geoLatitude, geoLongitude, 1); 


StringBuilder sb = new StringBuilder(); 


谨 判断 地 址 是 否 为 多 行 */ 
if (lstAddress.size() > 0) 


{ 
Address adsLocation = lstAddress.get(0); 


for(int 1=0;i<adsLocation.getMaxAddressLineIndex();i++) 
{ 

sb.append(adsLocation.getAddressLine(i)).append("\n"); 
} 
sb.append(adsLocation.getLocality()).append("\n"); 
sb.append(adsLocation.getPostalCode()).append("\n"); 
sb.append(adsLocation.getCountryName()); 


/* 

* 将 获取 的 地 址 

* 组 合 后 放 在 StringBuilder 对 象 中 输 昌 
/ 

strReturn = sb.toString(); 


A 


} 


catch(Exception e) 


{ 


e.printStackTrace(); 
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return strReturn; 


public Location getLocationProvider(Location Manager lm) 
{ 
Location retLocation = null; 
try 
{ 
Criteria mCriteria01 = new Criteria(); 
mCriteria01.setAccuracy(Criteria. ACCURACY _ FINE); 
ImCriteria01.setAltitudeRequired(false); 
mCriteria01.SetBearingRequired(false); 
mCriteria01.setCostAllowed(true); 
mCriteria01.setPowerRequirement(Criteria. POWER LOW); 
strLocationProvider = lm.getBestProvider(mCriteria0 1, true); 


retLocation = lIm.getLastKnownLocation(strLocationProvider); 
} 
catch(Exception e) 
{ 

mTextView01.setText(e.toString()); 

e.printStackTrace(); 


} 


return retLocation; 


private GeoPoint getGeoByLocation(Location location) 
{ 
GeoPoint gp = nmull; 
try 
{ 
席 当 位 置 存 在 */ 
if (location != null) 
{ 
double geoLatitude = location.getLatitude()*1E6; 
double geoLongitude = location.getLongitude()*1E6; 
gp =new GeoPoint((int) geoLatitude, (int) geoLongitude); 


} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 


return gp; 
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public static void refresh MapViewByGeoPoint 
(GeoPoint gp, MapView mv, int zoomLevel, boolean blIfSatellite) 
{ 

try 

{ 


myv.displayZoomControls(true); 

/* 取得 MapView 的 MapController */ 
MapController mc = myv.getController(); 
上 移 至 该 地 理 坐 标 地 址 */ 


mc.animateTo(gp); 


雍 放大 地 图 层级 */ 


mc.setZoom(zoomLevel); 


庆 设置 MapView 的 显示 选项 (卫星 、 街 道 )*/ 
if(bIfSatellite) 


{ 
myv.setSatellite(true); 
myv.setStreetView(true); 


} 


else 


{ 


myv.setSatellite(false); 


} 
} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 
} 


5) 定义 方法 processLocationUpdated(Location location)， 当 手机 收 到 位 置 更 改 时 ， 将 位 置 
传 入 GeoPoint 及 MapView。 上 有 具体 代码 如 下 。 


入 当 手 机 收 到 位 置 更 改 ， 将 位 置 传 入 GeoPoint 及 MapView */ 
private void processLocationUpdated(Location location) 


{ 
席 传 入 Location 对 象 ， 取 得 GeoPoint 地 理 坐 标 */ 
currentGeoPoint = getGeoByLocation(location); 


/* 更 新 MapView 显示 Google Map */ 
refresh MapViewByGeoPoint 
(currentGeoPoint, mMapView01, intZoomLevel true); 
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mTextView01.setText 


( 


getResources().getText(R.string.str my location).toString()+ 


"n"+getAddressbyGeoPoint(currentGeoPoint) 
); 
} 


(QOverride 
protected boolean isRouteDisplayed() 
{ 
return false; 
} 
} 


执行 后 将 显示 当前 位 置 的 定位 信息 。 如 图 9-19 所 示 ， 并 能 实现 及 时 更 新 功能 。 


example7 
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9.8 Google Translate APID 0 


ALT 


图 9-19 执行 效果 


在 Android 系统 中 ， 是 没有 手机 翻译 功能 的 。 但 是 Android 官方 为 其 提供 了 API 支持 ， 
通过 调用 Google Translate API 即 可 实现 翻译 功能 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 
实现 ， 介 绍 在 Android 中 通过 Google Translate API 实现 翻译 处 理 的 过 程 。 本 实例 的 源 代码 保 


存在 “光盘 :\daima\9\example8”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


9.8.1 Google Translate API 介 绍 


Google 的 在 线 翻 译 功 能 十 分 强大 ， 现 在 Google 已 经 开放 了 其 Ajax 的 API。 使 用 Ajax 技 


术 的 API， 可 以 仅 使 用 JavaScript 来 翻译 和 检测 网 页 


P 文 本 块 的 语言 。 此外， 可 以 在 网 页 的 任 


何 文本 字段 或 文本 区 域 启用 音译 。 例 如 ， 如 果 您 已 音译 为 北 印 度 语 ， 则 该 API 会 允许 用 户 使 
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用 英语 按照 发 音 拼 出 北 印 度 语 单词 ， 并 在 北 印 度 语 脚本 中 显示 出 
语言 API 旨 在 简单 容易 地 用 来 在 脱 机 翻译 不 可 用 时 翻译 和 检测 使 用 的 语言 。Google 公司 
计划 以 后 向 Ajax 语言 API 添加 更 多 令 人 激动 的 的 功能 ， 敬 请 期 竺 。 
google-api-translate-java 是 Java 语言 对 Google 翻译 引擎 的 封装 类 库 ， 具 体 使 用 方法 如 下 。 


import com.google.api.translate.Language; 
import com.google.api.translate.Translate; 


public class Main { 
public static void main(String[] args) { 

try { 

String translatedText = 
Translate.translate("Salut le monde", Language.FRENCH, Language.ENGLISH); 

System.out.printin(translatedText); 

} catch (Exception ex) { 
ex.printStackTrace(); 


9.8.2 具体 实现 


本 实例 的 主 文件 是 example8.java， 其 实现 流程 如 下 。 

第 1 步 : 在 Activity 中 设置 一 个 EditText 组 件 ， 用 于 接收 用 户 欲 翻译 的 字符 串 。 

第 2 步 : 编写 和 Google Translate API 通信 的 JavaScript， 并 以 HTML 格式 存储 在 “asssts” 
文件 夹 中 。 
第 3 步 : 在 HTML 网 页 中 编写 一 个 <a hre 他 的 链接 。 

第 4 步 : 当 用 户 在 EditText 输入 英文 后 ， 单 击 “ 中 文 ”链接 后 开始 翻译 处 理 ， 并 将 翻译 
结果 显示 在 WebView 中 。 

下 面 开始 讲解 文件 example8.java 的 具体 实现 流程 。 

1) 定义 WebSettings 对 象 webSettings， 用 于 获取 WebSettings。 有 具体 代码 如 下 。 


myEditTextl = (EditText) fndViewById(R.id.myEditTextl ); 
myWebViewl = (WebView) fmndViewById(R.id.myWebViewl); 


/# 取得 WebSettings */ 

WebSettings webSettings =ImyWebViewl.getSettings(); 
/* 设置 可 运行 JavaScript */ 
webSettings.setJavaScriptEnabled(true); 


webSettings.setSaveFormData(false); 
webSettings.setSavePassword(false); 
webSettings.setSupportZoom(false); 
myWebViewl.setWebChromeClient(new MyWebChromeClient()); 
入 设置 给 Html 调用 的 对 象 及 名 称 */ 
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myWebView1.addJavascriptInterface(new runJavaScript(), "irde"); 
/* 将 assets/google translate.html 载 入 */ 
String url = "file:///android asset/google translate.html"; 
myWebView!.loadUrl(ur)); 
} 


2) 定义 runJavaScript0 方 法 ， 用 于 调用 google translate.html 里 的 javaScript 以 显示 结果 。 


具体 代码 如 下 。 


final class runJavaScript 


{ 
public void runOnAndroidJavaScript() 
{ 
mHandler01.post(new Runnable() 
{ 
public void run() 
{ 
if (myEditTextl.getText().toString() (= "") 
{ 
/* 调用 google translate.html 里 的 javaScript */ 
myWebView!.loadUrl("javascript:translate("" 
+myEditTextl.getText().toString() + ")"); 
} 
} 
»); 
} 
} 


3) 定义 onJsAlert0 方 法 ， 用 于 捕捉 网 页 里 的 JavaScript 提示 语句 作为 js 调试 之 | 
至 LogCat 对 象 。 具 体 代 码 如 下 。 


Er 


/** 
* 捕捉 网 页 里 的 alert javascript 作为 js 调试 之 用 ， 并 输出 至 LogCat 
2 
final class MyWebChromeClient extends WebChromeClient 
{ 
(QOverride 


public boolean onJsAlert( WebView view, String url, 
String message, JsResult result) 
{ 
// TODO Auto-generated method stub 
Log.d(LOG TAG, message); 
// result.confirm(); 
return super.onJsAlert(view, url, message, result); 
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执行 后 的 效果 如 图 9-20 所 示 ; 当 输 入 英文 字符 并 单 击 “ 中 文 ” 链 接 后 ， 将 分 别 弹出 2 个 

对 话 框 , 分 别 如 图 9-21 和 图 9-22 所 示 ; 依次 单 击 “OK” 按 钮 , 即 可 实现 翻译 处 理 , 如 图 9-23 
所 示 显 示 的 翻译 结果 。 


昌 


example8 


©) The page at 'file://' says: 


Name 


mm 


图 9-20 ”初始 效果 图 9-21 单 击 “OK” 按 钮 


量 5554:ze 


呀 闭 便 3:07 AM 
©) The page at 'file://' says: examples | 


De 
中 文 
名 称 


9-23 ”翻译 结果 


图 9-22 单 击 “OK” 按 钮 
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通过 前 面 实例 的 学 习 ， 大 家 知道 Android 实现 GPS 导航 十 分 容易 。 实 际 上 除了 导航 
之 外 ， 还 可 以 在 地 图 上 绘制 线路 ， 计 算 距 离 ， 实 现 完整 的 导航 功能 。 在 本 节 的 内 容 中 ， 
将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 地 图 上 画图 并 计算 距离 的 具体 流程 。 本 实例 的 源 代 
码 保 存在 “光盘 :\daima\9\example9”， 下面 开始 讲解 本 实例 的 具体 实现 流程 。 
9.9.1 实现 原理 

Google Web Toolkit (GWT) 引入 了 JavaScript overlay 类 型 以 简化 将 整个 JavaScript 对 象 
家 族 集 成 到 GWT 项 目的 过 程 。 该 技术 有 很 多 优势 ， 如 利用 Java IDE 的 代码 完成 和 重 构 能 力 ， 
甚至 当 你 在 编写 无 类 型 的 JavaScript 对 象 时 也 可 以 充分 利用 这 一 优势 。 

通过 overlay 类 可 以 在 地 图 上 绘制 图 形 或 添加 图 片 ， 在 使 用 前 需要 引用 ， 格 式 如 下 。 


import com.google.android.maps.Overlay; 


在 本 实例 中 ， 设 置 了 一 个 继承 了 com.google.android.maps.Overlay 的 类 MyOverLay， 并 对 
方法 onDraw0 进 行 了 重 写 ， 这 样 实现 了 在 MapView 添加 轨迹 的 效果 。 
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9.9.2 具体 实现 


本 实例 的 主 文件 是 example9.java 和 OverLay.java， 下 面 分 别 介绍 其 实现 流程 。 
1. 文件 example8.java 
文件 examplel.java 的 功能 是 , 通过 自 定义 的 MyOveryLay 类 在 MapView 中 男 上 标记 。 文 
件 example8.java 的 具体 实现 流程 如 下 。 
1) 创建 MapView 对 象 , 分 别 对 象 初始 化 mTextView、mButton01、mButton02、mButton03 
和 mButton04。 上 基体 代 码 如 下 。 


/# 创建 MapView 对 象 */ 

mMapView = (MapView)findViewById(R.id.myMapView!); 
mMapController = mMapView.getController(); 

雍 对 象 初始 化 * 

mTextView = (TextView)findViewById(R.id.myText!); 
mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton02 = (Button)findViewById(R.id.myButton2); 
mButton03 = (Button)findViewById(R.id.myButton3); 
mButton04 = (Button)findViewById(R.id.myButton4); 


2) 设置 默认 的 放大 层级 为 17 级 ， 对 Provider 初始 化 处 理 并 分 别 获取 提供 者 与 位 置 ， 取 
得 当前 位 置 。 具 体 代码 如 下 。 
/# 设置 默认 的 放大 层级 */ 


ZoomLevel = 17; 


mMapController.setZoom(zoomLevel); 


Provider 初始 化 */ 

mLocation Manager = (Location Manager) 
getSystemService(Context.LOCATION SERVICE); 

上/ 取得 Provider 与 Location */ 

getLocationPrivider(); 

if(mLocation!=null) 


{ 
族 取得 目前 的 经 纬度 * 
gpl=getGeoByLocation(mLocation); 
gp2=gpl; 
/* 将 MapView 的 中 点 移 至 目前 位 置 %/ 
refresh MapView(); 
族 设置 事件 的 Listener */ 
mLocation Manager.requestLocationUpdates(mLocationPrivider, 

2000, 10, mLocationListener); 


} 


else 


{ 
new AlertDialog.Builder(example9.this).setTitle(" 系 统 信 息 ") 
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.SetMesSage(getResources(O.getString(R.string.str message)) 
.SetNegativeButton(" 人 确定 ",new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog, int which) 
{ 
example9 .this. finish(); 
} 
D) 
.Sshow(); 
} 


3) 定义 方法 mButton01.setOnClickListener()， 用 于 响应 单 击 “ 开 始 记 录 ” 按 钮 后 的 处 理 
事件 。 具 体 代码 如 下 。 


开始 记录 的 Button */ 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View V) 
{ 
gpl=gp2; 
上 清除 Overlay */ 
resetOverlay(); 
雍 画 起 点 * 
setStartPoint(); 
久 更 新 MapView */ 
refresh MapView(); 
人 # 重 设 移 动 距离 为 0， 并 更 新 TextView */ 
distance=0; 
mTextView.setText(" 移 动 距离 : 0M"); 
谍 启动 画 路 线 的 机 制 六 
_run=true; 
} 
); 


4) 定义 方法 mButton02.setOnClickListener()， 用 于 响应 单 击 “ 结 束 记 录 ” 按 钮 后 的 处 理 
事件 。 具 体 代码 如 下 。 


/# 结束 记录 的 Button */ 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View V) 
{ 
上 # 男 终点 */ 
setEndPoint(); 
/* 更 新 MapView */ 
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refresh MapView!(); 
谨 终止 画 路 线 的 机 制 六 
_run=false; 
} 
D); 


5) 定义 方法 mButton03.setOnClickListener()， 用 于 响应 单 击 


事件 。 有 具体 代码 如 下 。 
/# 缩小 地 图 的 Button */ 
mButton03.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View V) 
{ 
zoomLevel--; 
if(Zzoo0omLevel<1) 
{ 
ZoomLevel = 1; 
} 
mMapController.setZoom(zoomLevel); 
} 
); 


“缩小 地 图 ”按钮 后 的 处 理 


6) 定义 方法 mButton04.setOnClickListener()， 用 于 响应 单 击 
事件 。 具 体 代码 如 下 。 


放大 地 图 的 Button */ 
mButton04.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View V) 
4 
zoomLevel++; 
这 zoomLevel>mMapView.getMaxZoomLevel(O) 


{ 
ZoomLevel = mMapView.getMaxZoomLevel(); 


} 


mMapController.setZoom(zoomLevel); 


} 


“放大 地 图 ”按钮 后 的 处 理 


7) 定义 方法 onLocationChanged(Location location)， 用 于 监听 当前 位 置 的 变化 ， 如 果 变 化 


则 记 下 轨迹 线路 。 具 体 代 码 如 下 。 
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* MapView 的 Listener */ 
public final LocationListener mLocationListener = 
new LocationListener() 
{ 
(QOverride 
public void onLocationChanged(Location location) 
{ 
族 如 果 记 录 进 行 中 ， 就 画 路 线 并 更 新 移动 距离 */ 
if(_ run) 
{ 
雍 记 下 移动 后 的 位 置 */ 
gp2=getGeoByLocation(location); 
雍 画 路 线 */ 
setRoute(); 
人 更 新 MapView */ 
refresh MapView(); 
上 取得 移动 距离 */ 
distancet+=GetDistance(gp1,gp2); 
mTextView.setText(" 移 动 距 离 : "+format(distance)+"M"); 


gpl=gp2; 


Override 

public void onProviderDisabled(String provider) 
{ 

} 

(@Override 

public void onProviderEnabled(String provider) 
{ 


} 
(QOverride 


public void onStatusChanged(String provider,int status, 
Bundle extras) 


| 
} 
起 


8) 定义 方法 getGeoByLocation(Location location)， 用 于 取得 GeoPoint 的 方法 。 有 具体 代码 
如 下 。 


上 # 取得 GeoPoint 的 方法 */ 
private GeoPoint getGeoByLocation(Location location) 


{ 
GeoPoint gp = null; 
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try 
{ 
if (location != null) 
{ 
double geoLatitude = location.getLatitude()*1E6; 
double geoLongitude = location.getLongitude()*1E6; 
gp = new GeoPoint((int) geoLatitude, (int) geoLongitude); 


} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 


return gp; 


} 


9) 定义 方法 getLocationPrivider()， 用 于 获取 LocationProvider。 具 体 代码 如 下 。 


/* 取得 LocationProvider */ 

public void getLocationPrivider() 

{ 
Criteria mCriteria01 = new Criteria(); 
mCriteria01.setAccuracy(Criteria. ACCURACY FINE); 
ImCriteria01.setAltitudeRequired(false); 
mdCriteria01.setBearingRequired(false); 
ImCriteria01.setCostAllowed(true); 
ImCriteria01.SetPowerRequirement(Criteria. POWER LOW); 


mLocationPrivider = mLocation Manager 
.getBestProvider(mCriteria01, true); 
mLocation = mLocation Manager 
.getLastKnownLocation(mLocationPrivider); 


} 


10) 分 别 设置 起 点 方法 setStartPointOD、 路 线 的 方法 setRoute0、 终 点 的 方法 setEndPoint()、 
重 设 Overlay 的 方法 resetOverlay0 和 更 新 MapView 的 方法 setEndPoint()。 上 基体 代码 如 下 。 


久 设置 起 点 的 方法 */ 

private void setStartPoint() 

{ 
int mode=1]; 
OverLay mOverlay = new OverLay(gp1,gp2,mode); 
List<Overlay> overlays = mMapView.getOverlays(); 
overlays.add(mOverlay); 

} 

久 设置 路 线 的 方法 */ 


private void setRoute() 
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int mode=2; 
OverLay mOverlay = new OverLay(gp1l,gp2,mode); 
List<Overlay> overlays = mMapView.getOverlays(); 
overlays.add(mOverlay); 

} 

谍 设置 终点 的 方法 */ 

private void setEndPoint() 

{ 
int mode=3; 
OverLay mOverlay = new OverLay(gp1,gp2,mode); 
List<Overlay> overlays = mMapView.getOverlays(); 
overlays.add(mOverlay); 

} 

/* 重 设 Overlay 的 方法 */ 

private void resetOverlay() 


{ 
List<Overlay> overlays = mMapView.getOverlays(); 


overlays.clear(); 

} 

上 更 新 MapView 的 方法 */ 

public void refresh MapView!() 

{ 
mMapView.displayZoomControls(true); 
MapController myMC = mMapView.getController(); 
myMC.animateTo(gp2); 


myMC.setZoom(zoomLevel); 
mMapView.setSatellite(false); 
} 


11) 定义 方法 GetDistance(GeoPoint gpl,GeoPoint gp2)， 用 于 获取 两 点 间 的 距离 ， 并 通过 
去 format(double num) 处 理 移动 的 距离 。 具 体 代 码 如 下 。 
詹 取得 两 点 间 的 距离 的 方法 冯 
public double GetDistance(GeoPoint gp1,GeoPoint gp2) 
{ 


一 < 


方 


double Latlr = ConvertDegreeToRadians(gpl.getLatitudeE6()/1E0); 

double Lat2r = ConvertDegreeToRadians(gp2.getLatitudeE6()/1E0); 

double Longlr= ConvertDegreeToRadians(gpl.getLongitudeE6()/1E6); 

double Long2r= ConvertDegreeToRadians(gp2.getLongitudeE6()/1E6); 

谍 地 球 半 径 (KM) */ 

double R = 6371; 

double d = Math.acos(Math.sin(Lat1lr)*Math.sin(Lat2r)+ 
Math.cos(Lat1r)*Math.cos(Lat2r)* 
Math.cos(Long2r-Long1r))*R; 

return d*1000; 
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} 


private double ConvertDegreeToRadians(double degrees) 


{ 


} 


return (Math.PI180)*degrees; 


诬 format 移动 距离 的 方法 */ 
public String format(double num) 


{ 


NumberFormat formatter = new DecimalFormat("###"); 
String s=formatter.format(num); 


return s; 


(QOverride 
protected boolean isRouteDisplayed() 


{ 


} 


return false; 


本 实例 的 主 文件 是 example9.java 和 OverLay.java， 下 面 分 别 介 绍 其 实现 流程 。 
2. 文件 OverLay.java 


文件 OverLayjava 的 功能 是 ， 定 一 个 继承 自 
放 以 getProjection() 方 法 获取 Projection 对 象 ， 再 
法 将 getProjection() 方 法 转换 成 Point〈 点 )， 再 利用 Point 对 象 的 对 应 位 置 来 绘制 图 形 。 文 件 


绘制 图 形 ， 


OverLayjava 的 具体 代码 如 下 所 示 。 


public class OverLay extends Overlay 


{ 
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private GeoPoint gpl; 
private GeoPoint gp2; 
private int mRadius=6; 
private int mode=0; 


/# 构造 器 ， 传 入 起 点 与 终点 的 GeoPoint 与 mode */ 
public OverLay(GeoPoint gp1,GeoPoint gp2,int mode) 
{ 

this.gpl = gpl; 

this.gp2 = gp2; 

this.mode = mode; 


(@Override 
public boolean draw 


Overlay 的 子 类 OverLay， 并 在 MapView 上 


以 projection.toPixels(gp1, point) 方 


(Canvas canvas, MapView mapView, boolean shadow, long when) 


{ 

Projection projection = mapView.getProjection(); 

if (shadow 一 false) 

{ 
/* 设置 笔 刷 */ 
Paint paint = new Paint(); 
paint.setAntiAlias(true); 
paint.setColor(Color.BLUE); 


Point point = new Point(); 
projection.toPixels(gp1, point); 
们 inode=1: 创建 起 点 并 
if(mode==1) 
{ 
/* 定义 RectF 对 象 */ 
RectF oval=new RectF(point.x - mRadius, point.y - mRadius, 


point.x + mRadius, point.y + mRadius); 


此 绘制 起 点 的 圆 形 */ 
canvas.drawOval(oval, paint); 


} 

/mode=2: 画 路 线 */ 

else if{(mode==2) 

{ 
Point point2 = new Point(); 
projection.toPixels(gp2, point2); 


paint.setColor(Color.BLACK); 
paint.setStroke Width($5); 
paint.setAlpha(120); 
/* 画 线 */ 
canvas.drawLine(point.x, point.y, point2.x,point2.y, paint); 
} 
/*mode=3: 创建 终点 并 
else 1f{mode==3) 
{ 
/* 避免 误差 ， 先 画 最 后 一 段 的 路 线 */ 
Point point2 = new Point(); 


projection.toPixels(gp2, point2); 

paint.setStroke Width(5); 

paint.setAlpha(120); 

canvas.drawLine(point.x, point.y, point2.x,point2.y, paint); 


/* 定义 RectF 对 象 */ 
RectF oval=new RectF(point2.x - mRadius,point2.y - mRadius, 
point2.x + mRadius,point2.y + mRadius); 


/# 绘制 终点 的 圆 形 */ 
paint.setAlpha(255); 
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canvas.drawOval(oval, paint); 
} 
} 
return super.draw(canvas, mapView, shadow, when); 
} 
} 


执行 后 依次 单 击 “ 开 始 记录 ”和 “结束 记录 ”按钮 , 能 实现 GPS 轨迹 记录 。 效果 如 图 9-24 
所 示 。 
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图 9-24 ”执行 效果 
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在 现实 应 用 中 ， 二 维 条 码 的 应 用 比较 常见 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实 
现 ， 介 绍 在 Android 中 通过 swetake 实现 二 维 条 码 的 具体 流程 。 本 实例 的 源 代码 保存 在 “ 光 
得:\daima\9\example10”， 下 面 开 始 讲解 本 实例 的 具体 实现 流程 。 


9.10.1 ”实现 原理 


在 本 书 前 面 的 实例 中 ， 作 者 介绍 了 通过 Google Chat API 实现 二 维 条 码 的 流程 。 但 是 我 们 
不 能 保证 手机 都 处 于 联网 状态 ， 也 就 不 能 确保 使 用 Google ChatAPI 了 。 为 了 解决 这 个 问题 ， 
可 以 使 用 开放 的 Library， 例 如 最 常见 的 swetake， 读 者 可 以 去 http:/www.swetake.comy/ 网 站 下 
载 获 取 。 下 载 后 将 其 引入 到 Android 工程 中 ， 并 将 文件 名 称 改 为 SwetakeQRCode.jar。 


9.10.2 具体 实现 
本 实例 的 主 文件 是 examplel.java， 下 面 分 别 介绍 其 实现 流程 。 
1) 设置 应 用 程序 全 屏幕 运行 ， 而 不 使 用 标题 栏 。 具 体 代码 如 下 。 
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入 应 用 程序 全 屏幕 运行 ， 不 使 用 标题 栏 */ 
requestWindowFeature(Window.FEATURE NO TITLE); 
setContentView(R.layout.main); 


2) 取得 屏幕 解析 像素 ， 并 以 SurfaceView 作为 相机 预览 之 用 ， 绑 定 SurfaceView， 取 得 
SurfaceHolder 对 象 ， 并 产生 二 维 条 码 的 按钮 事件 处 理 。 有 具体 代码 如 下 。 


入 取得 屏幕 解析 像素 */ 
DisplayMetrics dm = new DisplayMetrics(); 


getWindowManager(0.getDefaultDisplay(O.getMetrics(dmy); 


ImTextView01l = (TextView) fndViewById(R.id.myTextViewl); 
ImTextView01.setText(R.string.str_qr_gen); 


/# 以 SurfaceView 作为 相机 Preview 之 用 */ 
mSurfaceView01 = (SurfaceView) fndViewById(R.id.mSurfaceViewl); 


/# 绑 定 SurfaceView， 取 得 SurfaceHolder 对 象 */ 
mSurfaceHolder01 = mSurfaceView01.getHolder(); 


上 六 Activity 必须 实现 SurfaceHolder.Callback */ 
msSurfaceHolder01.addCallback(examplel.this); 


庆 产生 QRCode 的 按钮 事件 处 理 */ 
mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton01.setOnClickListener(new Button.OnClickListener() 
{ 
(QOverride 
public void onClick(View arg0) 
{ 
/IODO Auto-generated method stub 
if(mEditText01.getText().toString()!="") 
{ 
访 传 入 setQrcodeVersion 为 4， 仅 能 接受 62 个 字符 *%/ 
Androi dQREncode(mEditText01.getText().toString(), 4); 
} 
} 
); 


mEditText01 = (EditText)findViewById(R.id.myEditText!); 
mEditTextO1.setText("DavidLanz"); 
mEditTextO1.setOnKeyListener(new EditText.OnKeyListener() 
{ 

(QOverride 

public boolean onKey(View v, int keyCode, KeyEvent event) 

{ 

// TODO Auto-generated method stub 
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return false; 


); 


3) 自 定义 产生 QRCode 的 方法 AndroidQREncode()， 实现 二 维 编 码 处 理 。 具 体 代 码 如 下 。 


访 自 定义 产生 QRCode 的 方法 党 
public void AndroidQREncode(String strEncoding, int qrcodeVersion) 


{ 
try 
{ 
履 建构 QRCode 编码 对 象 */ 
com.swetake.util.Qrcode testQrcode = 
new com.swetake.util.Qrcode(); 
LM',Q,H' */ 
testQrcode.setQrcodeErrorCorrect(MD); 
/#"N""A" 或 者 其 他 */ 
testQrcode.setQrcodeEncodeMode('B'); 
/* 0-20 */ 
testQrcode.setQrcodeVersion(qrcodeVersion); 
byte[] bytesEncoding = strEncoding.getBytes("utf-8"); 
if (bytesEncoding.length>0 && bytesEncoding.length <120) 
{ 
/* 将 字符 串通 过 calQrcode 方法 转换 成 boolean 数组 */ 
boolean[][] bEncoding =testQrcode.calQrcode(bytesEncoding); 
/# 依据 编码 后 的 boolean 数组 ， 绘 图 */ 
drawQRCode 
(bEncoding, getResources().getColor(R.drawable.black)); 
} 
} 
catch (Exception e) 
{ 
e.printStackTrace(); 
} 
} 


4) 定义 方法 drawQRCode0 ， 用 于 在 SurfaceView 上 绘制 QRCode 条 形 码 ， 解 锁 
SurfaceHolder 并 绘图 。 有 具体 代码 如 下 。 


/# 在 SurfaceView 上 绘制 QRCode 条 形 码 */ 
private void drawQRCode(boolean[][] bRect int colorFilD) 
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/* test Canvas*/ 
int intPadding = 20; 


上 # 欲 在 SurfaceView 上 绘图 ， 需 先 锁定 SurfaceHolder */ 
Canvas mCanvas01 = mSurfaceHolder01.lockCanvas(); 


上/# 设置 画布 绘制 颜色 */ 
mCanvas01.drawColor(getResources().getColor(R.drawable.white)); 


入 创建 画笔 % 
Paint mPaint01 = new Paint(); 


族 设置 画笔 颜色 及 模式 六 
mpPaint01.setStyle(Paint. Style.FILL); 
mPaint01.setColor(colorFill); 
ImPaint01.setStrokeWidth(1.0P); 


/# 逐一 加 载 2 维 boolean 数组 */ 
for (int i=0;i<bRectlength;i++) 


{ 
for (int j=0;j<bRect.length;j++) 
{ 
if (bRect[j][i]) 
{ 
此 依据 数组 值 ， 绘 出 条 形 码 方块 沁 
mCanvas01.drawRect 
( 
new Rect 
( 
intPadding+j*3+2， 
intPadding+i*3+2， 
intPaddingfj*3+2+3， 
intPadding+Ti*3+2+3 
), mPaint01 
); 
} 
} 


} 
/* 解锁 SurfaceHolder， 并 绘图 六 
mSurfaceHolder01.unlockCanvasAndPost(mCanvas01); 


} 


public void mMakeTextToast(String str, boolean isLong) 
{ 


iisLong 一 true) 


{ 
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Toast.makeText(examplel .this, str, Toast.LENGTH LONG).show(); 


} 


else 


{ 


Toast.makeText(examplel .this, str, Toast.LENGTH SHORT).show(); 


} 


(QOverride 
public void surfaceChanged 


(SurfaceHolder surfaceholder int format, int w, int h) 


{ 


} 


Log.i(TAG, "Surface Changed"); 


(QOverride 
public void surfaceCreated(SurfaceHolder surfaceholder) 


{ 


} 


Log.i(TAG, "Surface Changed'"); 


(QOverride 
public void surfaceDestroyed(SurfaceHolder surfaceholder) 


{ 


} 


Log.i(TAG, "Surface Destroyed"); 


执行 后 可 以 对 输入 的 文本 转换 为 二 维 条 形 码 。 如 图 9-25 所 示 。 
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在 手机 中 可 以 开发 一 个 二 维 码 的 扫描 程序 ， 这 样 可 以 随时 随地 解码 二 维 条 码 了 。 在 本 节 
的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 编写 动态 二 维 条 码 扫描 程序 的 具 
体 流 程 。 本 实例 的 源 代 码 保存 在 “光盘 :daimag\examplel11>， 下 面 开始 讲解 本 实例 的 具体 人 


现 流程 。 


9.11.1 实现 原理 

本 实例 同样 需要 使 用 第 三 方 开放 的 Library， 在 此 需要 引用 QRCode 项 目 ， 在 下 载 .jar 文 
件 之 后 ， 将 文件 名 修改 为 SourceForgeQRCode.jar， 并 导入 到 Android 工程 中 去 。 当 前 的 二 维 
码 标准 是 QRCode，QRCode 码 是 由 日 本 Denso 公司 于 1994 年 9 月 研制 的 一 种 矩阵 二 维 码 符 
号 ， 它 具有 一 维 条 码 及 其 他 二 维 条 码 所 具有 的 信息 容量 大 、 可 靠 性 高 、 可 表示 汉字 及 图 像 多 
种 文字 信息 、 保 密 防 伪 性 强 等 优点 。 


9.11.2 具体 实现 


本 实例 的 主 文件 是 example2.java， 下 面 分 别 介绍 其 实现 流程 。 
1) 分 别 创建 入 有 Camera 对 象 mCamera01，mButton01，mButton02 和 mButton03。 有 具体 
代码 如 下 。 


/* 创建 私有 Camera 对 象 */ 


private Camera mCamera01; 


private Button mButton01, mButton02, mButton03; 


2) 分 别 创建 变量 mImageView01、TAG、mSurfaceView01 和 mSurfaceHolder01 作为 
review 照 下 来 的 照片 使 用 ， 并 设置 默认 相机 预览 模式 为 false。 有 具体 代码 如 下 。 


匀 作为 review 照 下 来 的 相片 之 用 党 
private ImageView mImageView01; 
private String TAG = "HIPPO"; 
private SurfaceView mSurfaceView0!1; 


private SurfaceHolder mSurfaceHolder01; 


/# 默认 相机 预览 模式 为 false */ 


private boolean blfPreview = false; 


/** Called when the activity is first created. */ 


3) 设置 应 用 程序 全 屏幕 运行 ， 并 添加 红色 正方 形 方 框 View 供用 户 对 准 条 形 码 ， 然 后 将 
创建 的 红色 方 框 添加 至 此 Activity 中 。 有 具体 代码 如 下 。 


o 


(QOverride 
public void onCreate(Bundle savedInstanceState) 
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super.onCreate(savedInstanceState); 


旋 应 用 程序 全 屏幕 运行 ， 不 使 用 标题 栏 */ 
requestWindowFeature(Window.FEATURE NO TITLE); 


setContentView(R.layout.main); 


庆 添加 红色 正方 形 方 框 View， 供 用 户 对 准 条 形 码 */ 
DrawCaptureRect mDraw = new DrawCaptureRect 
( 
example2.this, 
110, 10, 100, 100, 
getResources().getColor(R.drawable.lightred) 
); 


族 将 创建 的 红色 方 框 添加 至 此 Activity 中 党 
addContentView 


( 


mDraw, 


new LayoutParams 


( 
LayoutParams.WRAP CONTENT, LayoutParams.WRAP CONTENT 


) 
p 


4) 分 别 取得 屏幕 解析 像素 ， 绑 定 SurfaceView 对 象 ， 设 置 预 览 大 小 。 具 体 代码 如 下 。 


庶 取得 屏幕 解析 像素 */ 
DisplayMetrics dm = new DisplayMetrics(); 


getWindowManager().getDefaultDisplay().getMetrics(dm); 


mlmageView01 = (ImageView) findViewById(R.idmyImageViewl); 


/# 以 SurfaceView 作为 相机 预览 之 用 */ 


mSurfaceView01 = (SurfaceView) fndViewById(R.id.mSurfaceViewl); 


/# 绑 定 SurfaceView， 取 得 SurfaceHolder 对 象 */ 
mSurfaceHolder01 = mSurfaceView01.getHolder(); 


Activity 必须 实现 SurfaceHolder.Callback */ 
msSurfaceHolder01.addCallback(example2.this); 


入 额外 的 设置 预览 大 小 设置 ， 在 此 不 使 用 * 
mSurfaceHolder01.setFixedSize(320, 240); 
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* 以 SURFACE TYPE PUSH BUFFERS(3) 

* 作为 SurfaceHolder 显示 类 型 

米 米 六 
mSurfaceHolder01.setType 
(SurfaceHolder.SURFACE TYPE PUSH BUFFERS); 


mButton01 = (Button)findViewById(R.id.myButton!1); 
mButton02 = (Button)findViewById(R.id.myButton2); 
mButton03 = (Button)findViewById(R.id.myButton3); 


5) 定义 方法 mButton01.setOnClickListener， 用 于 打开 相机 及 预览 二 维 条 形 码 。 


号 体 代 码 


如 下 。 


庆 打开 相机 及 预览 二 维 条 形 码 */ 
mButton01.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
public void onClick(View arg0) 


{ 


雍 自 定义 初始 化 打开 相机 方法 六 
initCamera(); 
} 
»); 


6) 定义 方法 mButton02.setOnClickListener， 用 于 停止 预览 。 具 体 代 码 如 下 。 


/和 停 正 预 星 47 
mButton02.setOnClickListener(new Button.OnClickListener() 
{ 

(@Override 

public void onClick(View arg0) 

{ 


旋 自 定义 重 置 相 机 ， 并 关闭 相机 预览 方法 */ 
resetCamera(); 
} 
); 


7) 定义 方法 mButton03.setOnClickListener， 用 于 拍照 QRCode 二 维 条 形 人 码 。 


基体 代码 


如 下 。 


庆 拍照 QRCode 二 维 条 形 码 */ 
mButton03.setOnClickListener(new Button.OnClickListener() 


{ 
(QOverride 
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public void onClick(View arg0) 
{ 
席 自 定义 拍照 方法 * 
takePicture(); 


); 
} 


8) 定义 方法 initCamera()， 用 于 自 定 义 初始 相机 方法 。 具 体 代 码 如 下 。 
/# 自 定义 初始 相机 方法 */ 


private void initCamera() 


{ 
if(!bIfPreview) 
{ 
雍 若 相 机 不 在 预览 模式 ， 则 打开 相机 % 
mCamera01 = Camera.open(); 
} 


if (mCamera01 != null && !blfPreview) 


{ 
Log.i(TAG, "inside the camera"); 


上 创建 Camera.Parameters 对 象 */ 


Camera.Parameters parameters = mCamera01.getParameters(); 


席 设置 相片 格式 为 了 了 EG */ 
parameters.setPictureFormat(PixelFormat.JPEG); 


# 指定 preview 的 屏幕 大 小 */ 
parameters.setPreviewSize(160, 120); 


庆 设置 图 片 分 辨 率 大 小 */ 


parameters.setPictureSize(160, 120); 


/六 将 Camera.Parameters (相机 参数 ) 设置 为 Camera */ 
mCamera01.setParameters(parameters); 


人/# setPreviewDisplay 唯一 的 参数 为 SurfaceHolder */ 
mCamera01.setPreviewDisplay(mSurfaceHolder01); 


上 # 立即 运行 Preview */ 


mCamera01.startPreview(); 
bIfPreview = true; 
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11) 定义 方法 onPictureTaken()， 对 传 入 的 图 片 进行 处 理 。 具 体 流程 如 下 。 


第 9 章 _ 绑 定 官方 的 服务 “二 
9) 定义 方法 takePicture0， 用 于 拍照 处 理 并 获取 图 像 。 有 具体 代码 如 下 。 


雍 拍照 获取 图 像 */ 
private void takePicture() 
{ 
if (mCamera01 != null && blfPreview) 
{ 
调用 takePicture0 方 法 拍照 六 
mCamera01.takePicture 
(shutterCallback, rawCallback, jpegCallback); 
} 
} 


定义 方法 resetCamera0， 实 现 相 机 重 置 ， 并 需要 释放 Camera 对 象 。 


局 重大 本 瘟 到 
private void resetCamera() 
{ 
if (mCamera01 != null && blfPreview) 
{ 
mCamera01.stopPreview(); 
/释放 Camera 对 象 */ 


//mCamera01.release(); 


mCamera01 = null; 


bIfPreview = false; 


private ShutterCallback shutterCallback = new ShutterCallback() 
{ 
public void onShutter() 


{ 


} 
瘟 


private PictureCallback rawCallback = new PictureCallback() 
{ 
public void onPictureTaken(byte[] _data, Camera camera) 
{ 
} 
总 


第 1 步 : 设置 onPictureTaken 传 入 的 第 一 个 参数 即 为 照片 的 byte。 
第 2 步 : 使 用 Matrix.postScale 方法 缩小 Bitmap Size。 


第 3 


步 : 创建 新 的 Bitmap 对 象 。 


L 体 代码 如 下 。 
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获取 4:3 图 片 的 居 
将 拍照 的 图 文件 以 ImageView 显示 出 来 。 
将 传 入 的 图 文件 译 码 成 字符 串 。 
定义 方法 mMakeTextToast 输出 提示 。 
引 体 代码 如 下 。 


红 人 色相 


THHI 
| 


部 分 100x100 像素 。 


private PictureCallback jpegCallback = new PictureCallback() 


{ 


public void onPictureTaken(byte[] _data, Camera camera) 


{ 
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try 


/# onPictureTaken 传 入 的 第 一 个 参数 即 为 相片 的 byte */ 
Bitmap bm = 
BitmapFactory.decodeByteArray(_data, 0， data.lengthb); 


int resizeWidth = 160; 

int resizeHeight = 120; 

float scaleWidth = ((float) resize Width) / bm.get Width(); 
float scaleHeight = ((float) resizeHeight) / bm.getHeight(); 


Matrix matrix = new Matrix(); 
访 使 用 Matrix.postScale 方法 缩小 Bitmap Size*/ 
matrix.postScale(scaleWidth, scaleHeight); 


地 


上 创建 新 的 Bitmap 对 象 *%/ 
Bitmap resizedBitmap = Bitmap.createBitmap 
(bm, 0, 0, bm.getWidth(), bm.getHeight(), matrix, true); 


入 获取 4:3 图 片 的 居中 红色 框 部 分 100x100 像素 */ 
Bitmap resizedBitmapSquare = Bitmap.createBitmap 
(resizedBitmap, 30, 10, 100, 100); 


/# 将 拍照 的 图 文件 以 ImageView 显示 出 来 */ 
mImageView01.setImageBitmap(resizedBitmapSquare); 


此 将 传 入 的 图 文件 译 码 成 字符 串 * 
String strQR2 = decodeQRImage(resizedBitmapSquare); 


if(strQR2!="") 

{ 
if (URLUtil.isNetworkUrl(strQR2)) 
{ 


谍 OMIA 规范 ， 网 址 条 形 码 ， 打 开 浏 览 器 上 网 */ 
mMakeTextToast(strQR2, true); 

Uri mUri = Uri.parse(strQR2); 

Intent intent = new Intent(Intent.ACTION VIEW, mUn); 


/* 


startActivity(intent); 
} 
else if(eregi("wtai://",strQR2)) 
{ 
放 OMIA 规范 ， 手 机 拨打 电话 格式 六 
String[] aryTemp01 = strQR2.split("wtai://"); 
Intent myIntentDial = new Intent 
( 
"android.intent.action.CALL,", 
Uri.parse("tel:"+aryTemp01[1]) 
); 
startActivity(myIntentDial); 
} 
else if(eregi("TEL:",strQR2)) 
{ 
谨 OMIA 规范 ， 手 机 拨打 电话 格式 */ 
String[] aryTemp01 = strQR2.split("TEL:"); 
Intent myIntentDial = new Intent 
( 
"android.intent.action.CALL,", 
Uri.parse("tel:"+aryTemp01[1]) 
); 
startActivity(myIntentDial); 
} 
else 
{ 
谍 若 仅 是 文字 ， 则 以 Toast 显示 出 来 */ 
mMakeTextToast(strQR2, true); 
} 


显示 图 文件 后 ， 立 即 重 置 相 机 ， 并 关闭 预览 * 


resetCamera(); 


/* 


再 重新 启动 相机 继续 预览 */ 


initCamera(); 


} 


catch (Exception e) 


{ 


Log.e(TAG, e.getMessage()); 


} 
所 


public void mMakeTextToast(String str, boolean isLong) 


{ 
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iisLong 一 true) 
{ 
Toast.makeText(example2.this, str, Toast.LENGTH LONG).show(); 
} 
else 
{ 
Toast.makeText(example2.this, str, Toast.LENGTH SHORT).show(); 
} 
} 


7IT 


12) 定义 方法 checkSDCard0， 判 断 记 忆 卡 是 否 存在 。 有 具体 代码 如 下 。 


private boolean checkSDCard() 

{ 
A# 判断 记忆 卡 是 否 存 在 % 
if(android.os.Environment.getExternalStorageState().equals 
(android.os.Environment.MEDIA MOUNTED)) 
{ 


return true; 


} 


else 


{ 


return false; 


} 


13) 定义 方法 decodeQRImage(Bitmap myBmp)， 用 于 解码 传 入 的 Bitmap 图 片 。 具 体 代 码 
如 下 。 


诺 解码 传 入 的 Bitmap 图 片 */ 


public String decodeQRImage(Bitmap myBmp) 
{ 
String strDecodedData = ""; 
try 
{ 
QRCodeDecoder decoder = new QRCodeDecoder(); 
strDecodedData =new String 
(decoder.decode(new AndroidQRCodeImage(myBmp))); 
} 


catch(Exception e) 


{ 


e.printStackTrace(); 


} 


return strDecodedData; 
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14) 自 定义 实现 二 维 条 码 图 像 类 AndroidQRCodeImage， 具 体 代码 如 下 。 


庆 自 定义 实现 QRCodeImage 类 闷 
class AndroidQRCodeImasge Implements QRCodeImage 
{ 


Bitmap image; 


public AndroidQRCodeImage(Bitmap image) 
\ 


this.image = image; 


public int getWidth() 


{ 
return image.getWidth(); 


} 


public int getHeight() 
{ 


return image.getHeight(); 


public int getPixel(int x, int y) 


{ 


return image.getPixel(x, y); 


} 
1$) 定义 类 DrawCaptureRect， 用 于 绘制 相机 预览 画面 里 的 正方 形 方 框 。 具 体 代码 如 下 。 


此 绘制 相机 预览 画面 里 的 正方 形 方 框 */ 
class DrawCaptureRect extends View 


{ 
private int colorFill; 
private int intLeft,intTop,intWidth,intHeight; 


Im 


public DrawCaptureRect 

( 
Context context, int intX, int intY, int intWidth, 
int intHeight, int colorFill 


super(context); 
this.colorFill = colorFill; 
this.intLeft = IntX; 
this.intTop = intY; 
this.intWidth = intWidth; 
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16) 定义 方法 eregi(String strPat, String strUnknow)， 实 现 自 定义 比较 


人 码 如 下 。 
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} 


/* 


public static boolean eregi(String strPat, String strUnknow) 


{ 


this.intHeight = intHeight; 


(QOverride 


protected void onDraw(Canvas canvas) 


{ 


Paint mPaint01 = new Paint(); 
mpPaint01.setStyle(Paint. Style.FILL); 
mpPaintO1.setColor(colorFill); 
mPaint01.setStrokeWidth(1.0F); 


此 在 画布 上 绘制 红色 的 


U 


条 方 边框 作为 瞄准 器 */ 


canvas.drawLine 


( 


this.intLeft, this.intTop, 


this.intLeft+intWidth, this.intTop, mPaintO1 


让 
canvas.drawLine 


( 


this.intLeft, this.intTop, 


this.intLeft, this.intToptintHeight, mPaint01 


); 
canvas.drawLine 


( 


this.intLeft+intWidth, this.intTop, 


this.intLeft+intWidth, this.intTop+intHeight, mPaint01 


Ds 
canvas.drawLine 


( 


this.intLeft, this.intTopt+intHeight, 


this.intLeft+intWidth, this.intTop+intHeight, mPaint01 


); 


super.onDraw(canvas); 


自 定义 比较 字符 串 方法 光 


String strPattern = "(?1)"+strPat; 
Pattern p = Pattern.compile(strPattern); 


Matcher m = p.matcher(strUnknow); 


returmn m.find(); 


字 


器 
日 


Ud 


处 理 


内 体 代 
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(QOverride 
public void surfaceChanged 
(SurfaceHolder surfaceholder, int format, int w, int h) 


{ 
Log.i(TAG, "Surface Changed"); 


} 


(QOverride 
public void surfaceCreated(SurfaceHolder surfaceholder) 
{ 
Log.i(TAG, "Surface Changed"); 
} 


(QOverride 
public void surfaceDestroyed(SurfaceHolder surfaceholder) 


{ 
Log.i(TAG, "Surface Destroyed"); 


} 


(QOverride 
protected void onPause() 


{ 


super.onPause(); 
} 
} 


执行 后 能 够 通过 手机 拍照 实现 二 维 码 解析 。 如 图 如 图 9-26 所 示 。 


图 9-26 ”执行 效果 


7 


在 现实 应 用 中 ， 可 以 设计 屏幕 的 显示 颜色 。 在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实 
现 ， 介 绍 在 Android 中 设置 手机 屏幕 颜色 的 具体 实现 流程 。 本 实例 的 源 代码 保存 在 “ 光 
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盘 :daimax9\example12”， 下 面 开 始 讲解 本 实例 的 具体 实现 流 和 


9.12.1 


实现 原理 


在 A 


ndroid 中 ， 可 以 通过 Android.os. 、 可 以 控制 


HHD 


| 手机 的 WakeLock (手机 的 锁 


机 制 )， 这 样 可 以 让 手机 的 屏幕 保持 在 恒 亮 状 态 ， 再 通过 程序 将 手机 亮度 调 到 最 高 255。 


9.12.2 


具体 实现 


本 实 


1. 文件 example12.java 


在 文 


牛 example12.java 中 ， 先 将 / 


例 的 主 文 件 是 example12.java 和 MyAdapter， 下 面 分 别 介绍 其 实现 流程 。 


屏幕 设置 为 全 屏 显 示 ， 然 后 以 PowerManager.newWake 


Lock0 方 法 来 获取 WakeLock 对 象 ， 并 记 下 Activity 启动 前 的 屏幕 亮度 。 当 启动 Activity 时 调 


用 onResume() 方 法 ， 


example3 
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诈 运 行 wakeLock() 方 法 ， 设 置 屏幕 亮度 为 255， 当 暂停 或 停止 Activity 
时 调用 onPause0， 并 运行 wakeUnlock 方法 ， 设 置 屏 幕 亮度 为 程序 启动 时 的 亮度 。 文 从 
.java 的 具体 代码 如 下 。 


iT 


public class example12 extends Activity 


{ 


private boolean lfLocked = false; 


private PowerManager.WakeLock mWakeLock; 
private PowerManager mPowerManager; 


private LinearLayout mLinearLayout; 


A 定义 menu 选项 标识 A 用 于 


识别 每 个 选项 的 事 


mul 
下 


上 */ 


static final private int M CHOOSE = Menu.FIRST; 
static final private int M_EXIT = Menu.FIRSTT+1L， 


入 颜色 与 文字 数组 */ 


private int[] color={R.drawable.white,R.drawable.blue, 
R.drawable.pink,R.drawable.green, 
R.drawable.orange,R.drawable.yellow}; 
private int[] text={R.string.str_white,R.string.str_blue, 


R.string.str pink,R.string.str_green, 


R.string.str_orange,R.string.str_ yellow}; 


(QOverride 


public void onCreate(Bundle savedInstanceState) 


{ 


super.onCreate(savedInstanceState); 


/* 必须 在 setContentView 之 


前 实现 回 屏幕 显示 */ 


requestWindowFeature(Window.FEATURE NO_TITLE); 


this.getWindow!().setFlags 
( 


WindowManager.LayoutParams.FLAG FULLSCREEN, 
WindowManager.LayoutParams.FLAG FULLSCREEN 


» 


setContentView(R.layout.main); 


/* 在 Activity 启动 时 将 屏幕 亮度 调整 为 最 亮 
由 
WindowManager.LayoutParams lp = getWindow().getAttributes(); 


lp.screenBrightness = 1.0f; 
getWindow().setAttributes(lp); 


/* 初始 化 mLinearLayout */ 
mLinearLayout=(LinearLayout)findViewById(R.id.myLinearLayout1); 


/* 取得 PowerManager */ 

mPowerManager = (PowerManager) 
getSystemService(Context.POWER SERVICE); 
/* 取得 WakeLock */ 

mWakeLock = mPowerManager.newWakeLock 


( 
PowerManager.SCREEN BRIGHT WAKE LOCK, "BackLight" 


下 


(@Override 
public boolean onCreateOptionsMenu(Menu menu) 


/x* menu 群 组 ID */ 

int idGroup1l = 0; 

/* menultemID */ 

int orderMenuIteml = Menu.NONE; 

int orderMenuItem2 = Menu.NONET+1; 

/* 建立 menu 头 
menu.add(idGroup1,M_CHOOSE,orderMenuItem1l,R.string.str_title); 
menu.add(idGroup1,M_EXIT,orderMenuItem2,R.string.str_exit); 
Imenu.SetGroupCheckable(idGroupl, true, true); 


return super.onCreateOptionsMenu(menu); 


(@Override 


public boolean onOptionsItemSelected(Menultem item) 


switch(item.getltemld()) 
{ 
case (M CHOOSE): 
/# 弹出 选择 背后 颜色 的 AlertDialog */ 
new AlertDialog.Builder(example12.this) 


.setTitle(getResources().getString(R.string.str_title)) 
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.setAdapter(new MyAdapter(this,color,text),listener1) 
.setPositiveButton(" 取 消 "， 
new DialogInterface.OnClickListener() 
{ 
public void onClick(DialogInterface dialog, int which) 
{ 
} 


D) 
.Sshow(); 
break; 
case (M EXIT): 
局 久 并 柱 / 末 +/ 
this.finish(); 
break; 


} 


return super.onOptionsItemSelected(item); 


} 


人 # 选择 背后 颜色 的 AlertDialog 的 OnClickListener */ 
OnClickListener listenerl=new DialogInterface.OnClickListener() 


{ 
public void onClick(DialogInterface dialog,int which) 
{ 
上 # 更 改 背 景 颜 色 * 
mLinearLayout.setBackgroundResource(color[which]); 
/* 通过 Toast 提示 显示 设 定 的 颜色 */ 
Toast.makeText(example3.this, 
getResources().getString(text[which]), 
Toast.LENGTH LONG).show(); 
} 
上 
(QOverride 
protected void onResume() 
{ 
/* onResume() 重 启 时 运行 wakeLock0 方 法 */ 
wakeLock(); 
super.onResume(); 
} 
(@Override 
protected void onPause() 
{ 
/* onPause() 暂 停 时 运行 wakeUnlock0) 方 法 */ 
wakeUnlock(); 
super.onPause(); 
} 
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/# 唤起 WakeLock 的 方法 */ 
private void wakeLock() 
{ 

if (lifLocked) 

{ 


ifLocked = true; 
mWakeLock.acquire(); 
} 
} 


/* 释放 WakeLock 的 方法 */ 
private void wakeUnlock() 
{ 
1f (ifLocked) 
{ 
mWakeLock.release(); 
ifLocked = false; 
} 
} 
} 


2. 文件 MyAdapter.java 
在 文件 MyAdapterjava 中 ， 设 置 了 背景 颜色 菜单 Adapter 继承 来 自 android.widget. 
BaseAdapter， 使 用 change_color.xml 作为 Layout， 具 体 代 码 如 下 。 


package irdc.example12; 


A 

import irdc.example12.R; 

import android.content.Context; 
import android.view.LayoutInflater; 
import android.view.View; 

import android.view.ViewGroup; 
import android.widget.BaseAdapter; 
import android.widget.TextView; 


/* 自 定 义 的 Adapter， 继 承 android.widget.BaseAdapter */ 
public class MyAdapter extends BaseAdapter 
{ 


private LayoutInflater mInflater; 


private int[] color; 
private int[] text; 


public MyAdapter(Context context,int[] _color,int[] _text) 


{ 
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minflater = LayoutInflater.from(context); 
color= color; 
text= text; 


} 


/* 继承 BaseAdapter， 重 写 履 六 方法 */ 


(QOverride 
public int getCount() 


{ 


return text.length; 


} 


(QOverride 
public Object getItem(int position) 
{ 


return text[position|; 


} 


(@Override 
public long getItemld(int position) 
{ 

return position; 


} 


(@Override 
public View getView(int position, View convertView,ViewGroup par) 


{ 
ViewHolder holder:; 


if(convertView == null) 

{ 
/* 使 用 自 定义 的 change_color 实现 布局 */ 
convertView = mlInflater.inflate(R.layout.change color, null); 
/* 初始 化 holder 的 文字 */ 
holder = new ViewHolder(); 
holder.mText=(TextView)convertView.findViewById(R.id.myText); 
convertView.setTag(holder); 


} 


else 


{ 
holder = (ViewHolder) convertView.getTag(); 


} 


holder.mText.setText(text[position]); 


holder.mText.setBackgroundResource(color[position]); 


return convertView; 
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/* class ViewHolder */ 
private class ViewHolder 


{ 


TextView mText; 


} 
} 


执行 后 将 按照 默认 样式 显示 屏幕 颜色 ， 如 图 9-27 所 示 ; 单 击 “MENU” 后 弹出 2 个 选项 
卡 ， 如 图 9-28 所 示 ; 单 击 “选择 背光 颜色 ”选项 后 弹出 设置 对 话 框 ， 在 此 可 以 设置 要 显示 颜 
色 ， 如 图 9-29 所 示 。 


选择 背光 颜色 


图 9-27 默认 效果 
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图 9-29 设置 对 话 框 
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除了 拨打 电话 和 发 送 短信 外 ， 智 能 手机 一 般 还 具备 音频 /视频 播放 、 移 动 上 网 、 蓝 牙 、 收 
音 机 、 软 件 下 载 及 游戏 等 功能 。 特 别 是 游戏 功能 的 强大 与 否 ， 直 接 影响 了 手机 设备 的 销量 。 
在 本 章 的 内 容 中 ， 将 详细 讲解 Android 手机 游戏 开发 的 基本 流程 ， 并 通过 几 个 典型 实例 的 实 
现 ， 详 细 介绍 几 个 典型 Android 手机 游戏 应 用 的 具体 开发 流程 。 


]0.1 Graphics 口 口 口 口 


在 手机 上 开发 娱乐 游戏 ， 首 先 要 实现 绘图 功能 。 在 Android 系统 中 ， 绘 图 功能 是 通过 
Graphics 类 实现 的 。Graphics 类 能 够 很 方便 地 绘制 2D 图 像 ， 并 填充 颜色 。 在 本 书 的 前 几 章 
实例 中 ， 已 经 涉及 了 此 类 的 使 用 知识 ， 在 本 节 的 内 容 中 ， 将 进一步 剖析 Graphics 类 的 基本 
知识 。 


10.1.1 _ Color 类 


Color 类 即 Android.Graphics.Color， 在 Android 平台 上 表示 颜色 的 方法 有 很 多 种 ，Color 
提供 了 常规 主要 颜色 的 定义 ,比如 ColorBLACK 和 ColorGREEN 等 , 平时 创建 时 主要 使 用 以 
下 静态 方法 。 

1) static int argb(int alpha, int red, int green, int blue): 构造 一 个 包含 透明 对 象 的 颜色 。 

2) static int rgb(int red, int green int blue): 构造 一 个 标准 的 颜色 对 象 。 

3 ) static int parseColor(String colorString): 解析 一 种 颜色 字符 串 的 值 ， 比 如 传 入 
ColorBLACK。 

本 类 返回 的 均 为 一 个 整 型 ， 如 ， 绿 色 为 0xff00fFf00， 红 色 为 0xffff0000。 可 以 将 这 个 
DWORD 型 看 做 AARRGGBB，AA 代表 Aphla 透明 色 ， 后 面 的 就 不 难 理解 ， 每 个 部 分 的 取 
值 范围 是 0 一 255。 


10.1.2 ”Paint 类 

Paint 类 即 Android.Graphics.Paint， 我 们 可 以 理解 Paint 类 为 画笔 、 画 刷 的 属性 定义 ， 本 类 
中 常用 的 方法 如 下 。 

1) voidtresetO: 重 置 。 

2) void setARGB(int a, int x, int g, intb) 或 void setColor(int color): 用 于 为 设置 Paint 对 象 的 
颜色 。 

3 ) void setAntiAlias(boolean aa): 用 于 抗 锯齿 。 需 要 配合 void setFlags (PaintANTI_ 
ALIAS FLAG) 来 帮助 消除 锯齿 ， 使 其 边缘 更 平滑 。 
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4) Shader setShader(Shader shader): 用 于 设置 阴影 。Shader 类 是 一 个 矩阵 对 象 ， 如 果 为 


NULL， 将 清除 阴影 。 


5) void setStyle(Paint.Style style): 用 于 设置 样式 。 一般 为 FILL (填充 ) 或 者 STROKE (四 


陷 ) 效果 。 
6) void setTextSize(float textSize): 设置 字体 


大 小 。 


7) void setTextAlign(Paint.Align align): 文本 对 齐 方式 。 


8) Typeface setTypeface(Typeface typeface): 
内 部 的 字体 ， 对 于 中 文 来 说 一 般 为 宋体 ， 还 可 以 


设置 字体 ， 通 过 Typeface 可 以 加 载 Android 
自己 添加 字体 ， 例 如 雅 黑 等 。 


9) void setUnderlineText(boolean underlineText): 用 于 设置 下 画 线 ， 需 要 结合 void setFlags 


(PaintUNDERLINE TEXT FLAG) 方 法 来 实现 。 


下 面 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 Color 类 和 Paint 类 实现 绘图 处 理 的 
流程 。 本 实例 的 功能 是 绘制 一 个 矩形 ， 源 代码 保存 在 “光盘 :daima\l0vexample1”， 下 面 开始 


讲解 本 实例 的 具体 实现 流程 。 
第 一 步 : 编写 主 文件 main.xml。 


A CI 


第 二 步 : 编写 文件 Activityjava， 通 过 “mGameView = new GameView(this)”， 用 Activity 
类 的 setContentView 方法 来 设置 要 显示 的 具体 View 类 。 文 件 Activityjava 的 具体 实现 代码 如 下 。 


package com.examplel; 


import Android.app.Activity; 
import Android.os.Bundle; 


public class Activity01 extends Activity 


{ 


private GameView mGameView; 

/** Called when the activity is first created. 
(@Override 

public void onCreate(Bundle savedInstance, 


{ 


super.onCreate(savedInstanceState); 


State) 


mGameView = new GameView(this); 


setContentView(mGameView); 


} 


AAA 一 


第 三 步 : 编写 主 文件 draw.java， 其 功能 是 绘制 出 指定 的 图 形 。 具 体 实现 流程 如 下 。 
1) 声明 Paint 对 象 mPaint， 定 义 draw 分 别 用 于 构建 对 象 和 开启 线程 。 具 体 代码 如 下 。 


/* 声明 Paint 对 象 */ 
private Paint mPaint = null; 


画 国 531 


Android 开发 完全 实战 宝典 


public draw(Context context) 


{ 
super(context); 
和 # 构建 对 象 % 
mpPaint = new Paint(); 
席 开启 线程 */ 
new Thread(this).start(); 
} 


2) 定义 方法 onDraw， 先 设置 Paint 格式 和 颜色 ， 
和 属性 实现 绘制 处 理 。 具 体 代 码 如 下 。 


让 
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public void onDraw(Canvas canvas) 


{ 


super.onDraw(canvas); 


/# 设置 Paint 为 无 锯齿 */ 
mPaint.setAntiAlias(true); 


/* 设置 Paint 的 颜色 */ 
mPaint.setColor(Color. WHITE); 
mPaint.setColor(Color.BLUE); 
mPaint.setColor(Color.YELLOW); 
mPaint.setColor(Color.GREEN); 

旋 同样 是 设置 颜色 * 
mPaint.setColor(Color.rgb(255, 0, 0)); 


全 提取 颜色 */ 
Color.red(O0xcccccc); 
Color.green(Oxcccccc); 


/* 设置 paint 的 颜色 和 Alpha 值 (arg,b) */ 
mPaint.setARGB(255, 255, 0, 0); 


放 设置 paint 的 Alpha 值 */ 
mPaint.setAlpha(220); 


/# 这 里 可 以 设置 为 男 外 一 个 paint 对 象 */ 
mPaint.set(new Paint()); 
族 设置 字体 的 尺寸 所 
mPaint.setTextSize(14); 


根据 提取 的 颜色 、 尺 寸 、 风 格 、 字 体 
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/* 设置 paint 的 风格 为 “空心 ”对 
谨 当然 也 可 以 设置 为 “实心 ”(Paint.Style.FILL)*/ 
mPaint.setStyle(Paint.Style.STROKE); 


访 设置 “空心 ”的 外 框 的 宽度 */ 
mPaint.setStroke Width(5); 


TH 


人 # 得 到 Paint 的 一 些 属 性 六 
Log.i(TAG, "paint 的 颜色 : "+mPaint.getColorO); 


Log.i(TAG, "paint 的 Alpha: "+ mPaint.getAlpha()); 


Log.i(TAG, "paint 的 外 框 的 宽度 : "+ mPaint.getStrokeWidth(); 


Log.i(TAG, "paint 的 字体 尺寸 : "+ mPaint.getTextSizeO); 


订 绘制 二 个 惩 形 洛 
上 # 肯定 是 一 个 空心 的 矩形 */ 


canvas.drawRect((320 - 80) /2, 20, (320 - 80) /2 + 80, 20 + 40, mPaint); 


谨 设置 风格 为 实心 Y 
mPaint.setStyle(Paint.Style.FILL); 


mPaint.setColor(Color.GREEN); 


绘制 绿色 实心 矩形 */ 
canvas.drawRect(0, 20, 40, 20 + 40, mPaint); 
} 


Tt 


小 ql 
Hl 
廊 


3) 分 别 定义 触 屏 事件 onTouchEvent， 按 键 按 下 事件 onKeyDown， 按 键 弹 起 
onKeyUp。 具 体 代 码 如 下 。 
入 触 屏 事件 */ 


public boolean onTouchEvent(MotionEvent event) 


{ 


return true; 


/# 按键 按 下 事件 */ 
public boolean onKeyDown(int keyCode, KeyEvent event) 


{ 


return true; 


/* 按键 弹 起 事件 
public boolean onKeyUp(int keyCode, KeyEvent event) 
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return false; 
} 
public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) 
{ 
return true; 
} 
public void run() 
{ 
while (!'Thread.currentThread().isInterrupted()) 
{ 
try 
{ 
Thread.sleep(100); 
} 
catch (InterruptedException e) 
Thread.currentThread().interrupt(); 
) 
/# 使 用 postInvalidate 可 以 直接 在 线程 中 更 新 界面 */ 
postInvalidate(); 
} 
} 


} 
执行 后 的 效果 如 图 10-1 所 示 。 


10.1.3 Canvas 
Canvas 类 Canvas 名 为 画布 ， 可 以 看 作 是 一 种 处 理 过 程 ， 使 用 各 种 方法 来 管理 Bitmap、 

GL 或 者 Path 路 径 ， 同 时 它 可 以 配合 Matrix 和 矩阵 类 给 图 像 做 旋转 、 缩 放 等 操作 ， 同 时 Canvas 

类 还 提供 了 裁剪 、 选 取 等 操作 


ii} 


o 
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下 面 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 使 用 Canvas 类 的 流程 。 本 实例 的 源 
代码 保存 在 “光盘 :daima\10\example2”， 其 主 文件 是 example2.java， 具 体 代码 如 下 。 


public class example2 extends View implements Runnable 


{ 
/* 声明 Paint 对 象 */ 
private Paint mPaint = null; 


public example2(Context context) 


{ 
super(context); 
1* 构建 对 象 光 
mPaint = new Paint(); 
庆 开启 线程 * 
new Thread(this).start(); 
} 


public void onDraw(Canvas canvas) 


{ 


super.onDraw(canvas); 


席 设置 画布 的 颜色 */ 
canvas.drawColor(Color.BLACK); 


族 设置 取消 锯齿 效果 */ 
mPaint.setAntiAlias(true); 


人 设置 裁剪 区 域 */ 
canvas.clipRect(10, 10, 280, 260); 


席 先 锁定 画布 *% 
canvas.save(); 

雍 旋转 画布 岂 
canvas.rotate(45.0f); 


谍 设置 颜色 及 绘制 矩形 */ 
mpPaint.setColor(Color.RED); 
canvas.drawRect(new Rect(15,15,140,70), mPaint); 


入 解除 画布 的 锁定 */ 


canvas.restore(); 


证 设置 颜色 及 绘制 另 一 个 矩形 */ 
mpPaint.setColor(Color.GREEN); 
canvas.drawRect(new Rect(150,75,260,120), mPaint); 
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} 


入 触 屏 事件 * 
public boolean onTouchEvent(MotionEvent event) 


{ 


return true; 


} 


/# 按键 按 下 事件 */ 
public boolean onKeyDown(int keyCode, KeyEvent event) 


return true; 


入 按键 弹 起 事件 当 
public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 


return false; 


public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) 


{ 
return true; 
) 
public void run() 
{ 
while (!'Thread.currentThread().isInterrupted()) 
{ 
try 
{ 
Thread.sleep(100); 
} 
catch (InterruptedException e) 
{ 
Thread.currentThread().interrupt(); 
} 
/* 使 用 postInvalidate 可 以 直接 在 线程 中 更 新 界面 */ 
postInvalidate(); 
} 
} 


} 


执行 后 的 效果 如 图 10-2 所 示 。 
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example2 


图 10-2 执行 效果 


10.1.4 Rect 类 

Rect 类 即 Android.Graphics.Rect， 可 以 理解 为 矩形 区 域 , Rect 类 除了 表示 一 个 矩形 区 域 位 
置 描述 外 ,Android 提示 主要 可 以 帮助 用 户 计 算 图 形 之 间 是 否 有 碰撞 (包含 ) 关 系 ,对 于 Android 
游戏 开发 比较 有 用 ， 其 主要 的 成 员 contains 有 如 下 3 种 重 载 方 法 来 判断 包含 关系 。 


Ra 


boolean contains(int left, int top, int right, int bottom) 
boolean contains(int x, int y) 


boolean contains(Rect r) 


下 面 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 使 用 Canvas 类 的 流程 。 本 实例 的 源 
代码 保存 在 “光盘 :\daima\10\example3”， 其 主 文件 是 example.java， 具 体 代码 如 下 。 


/* 声明 Paint 对 象 */ 
private Paint mPaint = null; 


private example3 1 mGameView2 = null; 
public example(Context context) 
{ 

super(context); 

上 # 构建 对 象 */ 

mPaint = new Paint(); 


mGameView2 = new example3_ 1(context); 


雍 开启 线程 六 
new Thread(this).start(); 


} 
public void onDraw(Canvas canvas) 


{ 


super.onDraw(canvas); 


/* 设置 画布 为 黑色 省 


Wh 


pll 
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canvas.drawColor(Color.BLACK); 
此 取消 锯齿 */ 
mPaint.setAntiAlias(true); 


mpPaint.setStyle(Paint. Style.STROKE); 


和 # 定义 矩形 对 象 */ 
Rect rectl =new Rect(); 
谨 设置 矩形 大 小 */ 
rectl.left = 5; 

rectl] .top = 5; 
rectl.bottom = 25; 

rectl .right = 45; 


mPaint.setColor(Color.BLUE); 
谨 绘制 矩形 */ 
canvas.drawRect(rectl, mPaint); 


mPaint.setColor(Color.RED); 
/# 绘制 矩形 */ 
canvas.drawRect(50, 5, 90, 25, mPaint); 


mPaint.setColor(Color.YELLOW); 
/# 绘制 贺 形 (圆心 x, 圆心 y, 半 径 r,p) 
canvas.drawCircle(40, 70, 30, mPaint); 


入 定义 椭圆 对 象 */ 
RectF rectfl = new RectF(); 
旋 设置 椭圆 大 小 * 

rectfl .left = 80; 

rectfl .top = 30; 

rectfl .right = 120; 
rectfl.bottom = 70; 


mPaint.setColor(Color.LTGRAY); 
入 绘制 彬 圆 */ 
canvas.drawOval(rectf1, mPaint); 


人 # 绘制 多 边 形 */ 
Path pathl = new Path(); 


诺 设 置 多 边 形 的 点 */ 
pathl.moveTo(150+5, 80-50); 
pathl.lineTo(150+45, 80-50); 
pathl.lineTo(150+30, 120-50); 


上 
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pathl.lineTo(150+20, 120-50); 
谨 使 这 些 点 构成 封闭 的 多 边 形 */ 
pathl.close(); 


mPaint.setColor(Color.GRAY); 
庆 绘制 这 个 多 边 形 * 
canvas.drawPath(path1, mPaint); 


mPaint.setColor(Color.RED); 
mPaint.setStroke Width(3); 

入 绘制 直线 */ 

canvas.drawLine(S$, 110, 315, 110, mPaint); 


i 绘制 实心 几何 体 


mpPaint.setStyle(Paint.Style.FILL); 


{ 


和 # 定义 矩形 对 象 */ 
Rect rectl = new Rect(); 
诺 设置 矩形 大 小 */ 
rectl.left = 5; 

Tectl.top = 130+5; 
rectl.bottom = 130+25; 
rectl .right = 45; 


mPaint.setColor(Color.BLUE); 
谨 绘制 矩形 */ 
canvas.drawRect(rectl, mPaint); 


mPaint.setColor(Color.RED); 
/# 绘制 矩形 */ 
canvas.drawRect(50, 130+5, 90, 130+25, mPaint); 


mPaint.setColor(Color. YELLOW); 
/# 绘制 圆 形 ( 圆 心 x, 圆 心 y, 半 径 r,p) */ 
canvas.drawCircle(40, 130+70, 30, mPaint); 


入 定义 椭圆 对 象 */ 
RectF rectfl = new RectF(); 
旋 设置 椭圆 大 小 * 
rectfl.left = 80; 


rectfl .top = 130+30; 
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rectfl right = 120; 
rectfl .bottom = 130+70; 


mPaint.setColor(Color.LTGRAY); 
入 绘制 椭圆 */ 
canvas.drawOval(rectf1, mPaint); 


人 # 绘制 多 边 形 */ 
Path pathl = new Path(); 


诺 设 置 多 边 形 的 点 */ 
pathl.moveTo(150+5, 130+80-50); 
pathl.lineTo(150+45, 130+80-50); 
pathl.lineTo(150+30, 130+120-50); 
pathl.lineTo(150+20, 130+120-50); 
诺 使 这 些 点 构成 封闭 的 多 边 形 */ 
pathl.close(); 


mPaint.setColor(Color.GRAY); 
谨 绘制 这 个 多 边 形 * 
canvas.drawPath(path1, mPaint); 


mPaint.setColor(Color.RED); 

mPaint.setStroke Width(3); 

1 i 

canvas.drawLine($, 130+110, 315, 130+110, mPaint); 


访 通过 ShapeDrawable 来 绘制 几何 图 形 */ 
mGameView2.DrawShape(canvas); 


入 触 屏 事件 * 
public boolean onTouchEvent(MotionEvent event) 


{ 


return true; 


性 按键 按 下 事件 */ 
public boolean onKeyDown(int keyCode, KeyEvent event) 


{ 


return true; 


放 按键 弹 起 事件 
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public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 


return false; 


public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) 


{ 
return true; 
} 
public void run() 
{ 
while (!'Thread.currentThread().isInterrupted()) 
{ 
try 
{ 
Thread.sleep(100); 
】 
catch (InterruptedException e) 
{ 
Thread.currentThread().interrupt(); 
】 
/# 使 用 postInvalidate 可 以 直接 在 线程 中 更 新 界面 */ 
postInvalidate(); 
} 
} 


} 


执行 后 的 效果 如 图 10-3 所 示 。 
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10.1.5 NinePatch 类 


NinePatch 类 即 Android.Graphics.NinePatch，NinePatch 是 Android 平台 特有 的 一 种 非 矢 
量 图 形 自然 拉 伸 处 理 方法 ， 能 够 实现 在 拉 伸 常规 的 图 形 时 不 会 自动 缩放 的 效果 ， 实 例 中 
Android 开发 网 提示 大 家 对 于 Toast 的 显示 就 是 该 原理 ， 同 时 SDK 中 提供 了 一 个 工具 名 为 
Draw 9-Patch， 有 关 该 工具 的 使 用 可 以 参考 已 经 发 布 的 Draw 9-Patch 使 用 方法 。 由 于 该 类 提 
供 了 高 质量 支持 透明 的 缩放 方式 ， 所 以 图 形 格式 为 PNG， 文 件 命名 方式 为 .9.png 的 后 绥 比 
如 Android123.9.png。 


10.1.6 ”Matrix 类 

Matrix 类 即 Android.Graphics.Matrix， 能 够 实现 图 形 的 变换 操作 ， 例 如 常见 的 缩放 和 旋转 
处 理 。Matrix 中 有 关 图 形 的 变换 、 缩 放 等 相关 操作 常用 的 方法 有 如 下 几 种 。 

1) void reset(): 重 置 一 个 matrix 对 象 。 

2) void set(Matrix src): 复制 一 个 源 和 矩阵 ， 和 本 类 的 构造 方法 Matrix(Matrix src) 一 样 。 

3) boolean isIdentity0: 返回 这 个 矩阵 是 否定 义 〈 已 经 有 意义 )。 

4) void setRotate(float degrees): 指定 一 个 角度 以 (0,0〉 为 坐标 进行 旋转 。 

5 ) void setRotate(float degrees, float px, float py): 指定 一 个 角度 以 (px,py)〉 为 坐标 进行 
旋转 。 

6) void setScale(float sx, float sy): 缩放 处 理 。 

7) void setScale(float sx, float sy float px, float py): 以 坐标 (px,py) 进行 缩放 。 

8) void setTranslate(float dx, float dy): 平移 。 

9) void setSkew (float kx, float ky, float px, float py): 以 坐标 (px，py) 进行 倾斜 。 

10) void setSkew (float kx, float ky): 倾斜 处 理 
有 关 Matrix 类 实现 图 片 缩放 处 理 的 操作 实例 请 读者 参阅 本 书 的 4.17 内 容 。 


UT 


o 


10.1.7 Bitmap 类 
Bitmap 类 即 Android.Graphics.Bitmap, 是 一 个 位 图 操作 类 , 实现 对 位 图 的 基本 操作 。Bitmap 
中 提供 了 很 多 实用 的 方法 ， 其 中 最 为 常用 的 几 种 方法 总 结 如 下 。 
1 ) boolean compress(Bitmap.CompressFormat format, int quality OutputStream stream): 压 
缩 一 个 Bitmap 对 象 ， 根 据 相 关 的 编码 、 画 质保 存 到 一 个 OutputStream 中 。 其 中 第 一 个 参数 格 
式 目 前 有 JPG 和 了 PNG。 
2) void copyPixelsFromBuffer(Buffer src): 从 一 个 Buffer 缓冲 区 复制 位 图 像素 。 
3 ) void copyPixelsToBuffer(Buffer dst): 将 当前 位 图 像素 内 容 复制 到 一 个 Buffer 缓冲 区 。 
目前 在 Android 2.1 SDK 中 ， 创 建 位 图 对 象 包含 了 6 种 方法 。 当 然 使 用 的 API Level 均 为 
1， 所 以 说 从 Android 1.0 SDK 开始 就 支持 了 ， 大 家 可 以 放心 使 月 
4) 下 列 方法 用 于 创建 一 个 可 以 缩放 的 位 图 对 象 。 


A 


o 


static Bitmap createBitmap(Bitmap src) 
static Bitmap createBitmap(int[] colors, int width, int height, Bitmap.Config config) 
static Bitmap createBitmap(int[] colors, int offset, int stride, int width, int height, Bitmap.Config config) 
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static Bitmap createBitmap(Bitmap source, int x, int y, int width, int height, Matrix m, boolean filter) 
static Bitmap createBitmap(int width, int height, Bitmap.Config config) 

static Bitmap createBitmap(Bitmap source, int x, int y, int width, int height) 

static Bitmap createScaledBitmap(Bitmap src, int dstWidth, int dstHeight, boolean filter) 


5) final int getHeight0: 获取 高 度 。 

6) final int getWidth(): 获取 宽度 。 

7) final boolean hasAlpha0: 是 否 有 透明 通道 。 

8) void setPixel(int x, int y int color): 设置 某 像素 的 颜色 。 

9) int getPixel(int x, int y): 获取 某 像素 的 颜色 ， 此 处 返回 的 int 型 参数 是 指 color 的 定义 。 

在 本 书 4.18 内 中 ， 已 经 通过 具体 实例 介绍 了 Bitmap 类 实现 图 片 旋转 处 理 的 具体 流程 。 
下 面 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 使 用 Bitmap 类 实现 模拟 水 纹 效 果 的 

流程 。 本 实例 的 源 代 码 保存 在 “光盘 :daima\10\example4”， 其 主 文件 是 example4.java， 有 具体 

代码 如 下 。 


public class example4 extends View implements Runnable 
{ 

int BACKWIDTH:; 

int BACKHEIGHT:; 

short[] buf2; 

short[] bufl; 

int[] Bitmap2; 

int[] Bitmap!1; 

public example4(Context context) 


{ 


super(context); 


启 加 玉 国 所 字 

Bitmap image = BitmapFactory.decodeResource(this.getResources(),R.drawable.qq); 
BACKWIDTH = image.getWidth(); 

BACKHEIGHT = image.getHeight(); 


buf2 = new short[BACKWIDTH * BACKHEIGHT]; 
bufl = new short[BACKWIDTH * BACKHEIGHT]; 
Bitmap2 = new int[BACKWIDTH * BACKHEIGHT]; 
Bitmapl = new int[BACK WIDTH * BACKHEIGHT]; 


入 加 载 图 片 的 像素 到 数组 中 * 
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image.getPixels(Bitmap1, 0, BACKWIDTH, 0, 0, BACKWIDTH, BACKHEIGHT); 


new Thread(this).start(); 


void DropStone(int x,// x 坐标 
int yy 坐标 
int stonesize,// 波源 半径 
int stoneweighb/ 波源 能 量 


Wu 


{ 
for (int posx = x - stonesize; posx < x + stonesize; poOSX++) 
for (int posy = y - stonesize; posy <y + stonesize; posy++) 
if((posx- xX) * (posx - x) + (posy -y) * (posy - y) < stonesize * stonesize) 
bufl[BACKWIDTH * posy + posx| = (Short) -stoneweight; 
} 
void RippleSpread() 

{ 


for (inti= BACKWIDTH; i1<BACKWIDTH * BACKHEIGHT - BACKWIDTH,; i++) 


{ 
/ 波 能 扩散 
buf2[i] = (short) (((bufl[i - 1] + bufl[i + 1] + buflfi - BACKWIDTH] + bufl[i + 
BACKWIDTH]) >> 1) - buf2[i]); 
// 波 能 衰减 
buf2[i] -= buf2[i] >> 5; 


xX| 


/ 交换 波 能 数据 缓冲 
short[] ptmp = bufl; 
bufl = buf2; 

buf2 = ptmp; 


} 
诺 演 染 水 纹 效 果 六 
void render() 


{ 


int xoff, yoff; 

int k = BACKWIDTH,; 

for (inti= 1;i<BACKHEIGHT - 1; i++) 

{ 
for (intj = 0;]j < BACKWIDTH.; j++) 
{ 


~ 计算 1 用 移 量 */ 
xoff=bufl[k- 1|]-bufllk+ 1]; 
yoff = bufl[k - BACKWIDTH]| - bufl[k + BACKWIDTHI]; 
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/# 判断 坐标 是 否 在 窗口 范围 内 */ 


if ((i+ yoff) <0) 
{ 
Je 
continue; 
} 
if ((i+ yoff) > BACKHEIGHT) 
{ 
ke 
continue; 
} 
if (Qj + xoff) <0) 
{ 
k++; 
continue; 
} 
if (Gj + xoff) > BACKWIDTH) 
{ 
k++; 
continue; 
} 


入 计算 出 偏 移 像 素 和 原始 像素 的 内 存 地 址 1 
int pos1, pos2; 

posl = BACKWIDTH * (i + yoff) + (j + xoff); 
pos2 = BACKWIDTH * i+j; 
Bitmap2[pos2++] = Bitmap1[pos1++]; 

left: 


和 


public void onDraw(Canvas canvas) 


} 


super.onDraw(canvas); 


入 绘制 经 过 处 理 的 图 片 效果 光 


入 触 屏 事 件 */ 
public boolean onTouchEvent(MotionEvent event) 


{ 


return true; 
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canvas.drawBitmap(Bitmap2, 0, BACKWIDTH, 0, 0, BACKWIDTH, BACKHEIGHT, false, 
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} 


人 # 按键 按 下 事件 */ 
public boolean onKeyDown(int keyCode, KeyEvent event) 


{ 


return true; 


人 按键 弹 起 事件 节 
public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 


DropStone(BACKWIDTH/2, BACKHEIGHT/2, 10, 30); 
return false; 


public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) 


{ 


return true; 
} 
A 
* 线程 处 理 
Sh / 
public void run() 
{ 
while (!'Thread.currentThread().isInterrupted()) 
{ 
try 
{ 
Thread.sleep(50); 
} 
catch (InterruptedException e) 
{ 
Thread.currentThread().interrupt(); 
} 
RippleSpread(); 
render(); 
/# 使 用 postInvalidate 可 以 直接 在 线程 中 更 新 界面 */ 
postInvalidate(); 
} 
} 
} 
执行 后 将 通过 对 图 像 像 素 的 操作 数 来 模拟 水 纹 效果 ， 效 果 如 图 10-4 所 示 。 
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10.1.8 ” BitmapFactory 类 


BitmapFactory 类 即 Android.Graphics.BitmapFactory， 作 为 Bitmap 对 象 的 IO 类 ， 
BitmapFactory 类 提供 了 丰富 的 构造 Bitmap 对 象 的 方法 ， 比 如 从 一 个 字 节 数组 、 文 件 系 统 、 资 
源 ID 以 及 输入 流 中 创建 一 个 Bitmap 对 象 , 下 面 是 本 类 的 全 部 成 员 , 除了 decodeFileDescriptor 
外 ， 其 他 的 重 载 方法 都 很 常用 。 

(1) 从 字 节 数组 创建 


static Bitmap decodeByteArray(byte[] data, int offset, int length) 
static Bitmap decodeByteArray(byte[] data, int offset, int length, BitmapFactory.Options opts) 


《2) 从 文件 创建 ， 路 径 要 写 全 


static Bitmap decodeFile(String pathName, BitmapFactory.Options opts) 
static Bitmap decodeFile(String pathName) 


(3) 从 输入 流 句柄 创建 


static Bitmap decodeFileDescriptor(FileDescriptor fd, Rect outPadding, BitmapFactory.Options opts) 
static Bitmap decodeFileDescriptor(FileDescriptor fd) 


1 


(4) 从 Android 的 APK 文件 资源 中 创建 
Android 提示 是 从 “/res/” 的 drawable 中 创建 : 


static Bitmap decodeResource(Resources res, int id) 
static Bitmap decodeResource(Resources res, int id, BitmapFactory.Options opts) 
static Bitmap decodeResourceStream(Resources res, TypedValue value, InputStream is, Rect pad, 


BitmapFactory.Options opts) 
(5) 从 一 个 输入 流 中 创建 


static Bitmap decodeStream(InputStream 1s) 
static Bitmap decodeStream(InputStream is, Rect outPadding, BitmapFactory.Options opts) 
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10.1.9 ”Region 类 

Region 类 即 Android.Graphics.Region，Region 在 Android 平台 中 表示 一 个 区 域 ， 和 Rect 
不 同 的 是 ， 它 表示 的 是 一 个 不 规则 的 图 形 ， 可 以 是 椭圆 、 多 边 形 等 ， 而 Rect 仅仅 是 矩形 。 同 
样 Region 的 boolean contains(int x, int y) 成 员 可 以 判断 一 个 点 是 否 在 该 区 域内 。 


10.1.10 ”Typeface 类 

Typeface 类 即 Android.Graphics.Typeface, Typeface 类 帮助 描述 一 个 字体 对 象 ,在 TextView 
中 通过 使 用 setTypeface() 方 法 来 指定 一 种 输出 文本 的 字体 , 其 调用 成 员 create 方法 可 以 直接 指 
定 一 种 字体 名 称 和 样式 ， 例 如 ， 


Se 


static Typeface create(Typeface family, int style) 
static Typeface create(String familyName, int style) 


同时 使 用 isBold 和 isItalic 方法 可 以 判断 出 是 否 包含 粗 体 或 斜体 的 字形 。 


final boolean isBold() 
final boolean isItalic() 


除了 上 述 创建 方法 外 ， 该 类 还 可 以 从 APK 的 资源 或 从 一 个 具体 的 文件 路 径 获取 字体 ， 其 
具体 方法 如 下 。 


static Typeface createFromAsset(AssetManager mgr String path) 
static Typeface createFromFile(File path) 
static Typeface createFromFile(String path) 


10.1.11 Shader 类 

Shader 类 用 于 泻 染 图 像 和 一 些 几 何 图 形 。 下 面 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 
Android 中 使 用 Bitmap 类 实现 模拟 水 纹 效 果 的 流程 。 本 实例 的 源 代码 保存 在 “ 光 
担 :\daima\10\example5”， 其 主 文件 是 exampleS.java， 有 具体 代码 如 下 。 


上 


public class example5 extends View implements Runnable 


{ 
/* 声明 Bitmap 对 象 */ 
Bitmap mBitQQ =null; 
int BitQQwidth =0; 
int BitQQheight =0; 


Paint mPaint= null; 


/* Bitmap 泻 染 * 
Shader mBitmapShader = null; 


/# 线性 渐变 泻 染 */ 
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Shader mLinearGradient = null; 


/* 混合 泻 染 */ 
Shader mComposeShader = null; 


/# 放射 性 渐变 泻 染 */ 
Shader mRadialGradient = null; 


和 # 梯度 演 染 * 


Shader mSweepGradient = null; 


ShapeDrawable mShapeDrawableQQ = null; 


public example5(Context context) 


{ 


TileMode.MIRROR); 


DARKEN); 


REPEAT): 


super(context); 


席 加 载 资源 */ 
mBitQQ = ((BitmapDrawable) getResources().getDrawable(R.drawable.qq)).getBitmap(); 


入 得 到 图 片 的 宽度 和 高 度 汉 
BitQQwidth = mBitQQ.get Width(); 
BitQQheight = mBitQQ.getHeight(); 


/* 创建 BitmapShader 对 象 */ 
mBitmapShader = new BitmapShader(mBitQQ,Shader.TileMode. REPEAT,Shader. 


族 创建 线性 渐变 并 设置 渐变 的 颜色 数组 */ 

mLinearGradient = new LinearGradient(0,0,100,100, 

new int[] {Color.RED,Color.GREEN, Color.BLUE,Color. WHITE;}, 
null,Shader.TileMode.REPEAT); 

族 这 里 笔者 理解 为 “混合 演 染 ”-- 大 家 可 以 有 自己 的 理解 ， 能 明白 这 个 意思 就 好 */ 
mComposeShader = new ComposeShader(mBitmapShader, mLinearGradient,PorterDuff.Mode. 


席 构建 放射 性 渐变 对 象 ， 设 置 半径 的 属性 */ 

/* 这 里 使 用 了 BitmapShader 和 LinearGradient 进行 混合 */ 
当然 也 可 以 使 用 其 他 的 组 合 */ 

族 泥 合演 染 的 模式 很 多 ， 可 以 根据 自己 需要 来 选择 */ 
mRadialGradient = new RadialGradient(50,200,50, 

new int[]{Color.GREEN,Color.RED, Color.BLUE,Color.WHITE},null,Shader.TileMode. 


/# 构建 梯度 渐变 对 象 */ 
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mSweepGradient = new SweepGradient(30,30,new int[]{Color.GREEN,Color.RED,Color.BLUE, 
Color WHITE} ,null); 


mPaint = new Paint(); 


席 开启 线程 */ 
new Thread(this).start(); 


public void onDraw(Canvas canvas) 


{ 


super.onDraw(canvas); 


/将 图 片 裁剪 为 椭圆 形 
/# 构建 ShapeDrawable 对 象 并 定义 形状 为 椭圆 */ 
mShapeDrawableQQ = new ShapeDrawable(new OvalShape()); 


到 


访 设置 要 绘制 的 椭圆 形 的 东西 为 ShapeDrawable 图 片 */ 
mShapeDrawableQQ.getPaint().setShader(mBitmapShader); 


上 # 设置 显示 区 域 */ 
mShapeDrawableQQ.setBounds(0,0, BitQQwidth, BitQQheight); 


/* 绘制 ShapeDrawableQQ */ 
mShapeDrawableQQ.draw(canvas); 


诺 绘制 渐变 的 矩形 */ 
mpPaint.setShader(mLinearGradient); 
canvas.drawRect(BitQQwidth, 0, 320, 156, mPaint); 


* 显示 混合 演 染 效果 */ 
mPaint.setShader(mComposeShader); 
canvas.drawRect(0, 300, BitQQwidth, 300+BitQQheight, mPaint); 


此 绘制 环形 渐变 */ 
mpPaint.setShader(mRadialGradient); 
canvas.drawCircle(50, 200, 50, mPaint); 


此 绘制 梯度 渐变 */ 
mPaint.setShader(mSweepGradient); 
canvas.drawRect(150, 160, 300, 300, mPaint); 


入 触 屏 事件 */ 
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public boolean onTouchEvent(MotionEvent event) 


{ 


return true; 


* 按键 按 下 事件 
public boolean onKeyDown(int keyCode, KeyEvent event) 


{ 


return true; 


人 # 按键 弹 起 事件 *% 
public boolean onKeyUp(int keyCode, KeyEvent event) 


return false; 


典型 手机 游戏 应 用 


public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) 


J 


{ 
return true; 
} 
7 米 米 
* 线程 处 理 
uh 
public void run() 
{ 
while (!'Thread.currentThread().isInterrupted()) 
{ 
try 
{ 
Thread.sleep(100); 
} 
catch (InterruptedException e) 
{ 
Thread.currentThread().interrupt(); 
} 
从 使 用 postInvalidate 可 以 直接 在 线程 中 更 新 界 国 
postInvalidate(); 
} 
} 
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执行 后 的 效果 如 图 10-5 所 示 。 


> 园 古 交合 6:48m 


example5 


图 10-5 执行 效果 


10.2 0000 


大 多 数 知 名 企业 的 游戏 框架 都 是 基于 Apache Struts、Spring 和 Hibernate 等 开发 框架 的 。 
这 些 框架 都 是 基于 MVC 设计 模式 ， 业 务 和 逻辑 被 分 开 ， 这 已 经 成 为 当前 游戏 开发 的 主流 模 
式 。 在 本 节 的 内 容 中 ， 作 者 将 简单 分 析 Android 平台 提供 的 View 和 Surfaceview 类 来 作为 
MVC 中 视图 基 类 的 开发 框架 。 


10.2.1 View 类 

在 下 面 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ,介绍 在 Android 中 使 用 View 类 实现 屏幕 
更 新 显示 的 流程 。 本 实例 的 源 代 码 保 存在 “光盘 :\daima\l0\example7”， 其 主 文件 是 
example7.java， 具 体 代 码 如 下 。 


public class example7 extends View 
{ 
int miCount = 0; 
int y=0; 
public example7(Context context) 
{ 
super(context); 
} 
/* 绘图 处 理 */ 
public void onDraw(Canvas canvas) 


{ 


if (miCount < 100) 
{ 


IICount++; 
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miCount = 0; 


了 了 


上 绘图 */ 


Paint mPaint = new Paint(); 


switch (miCount%4) 


{ 


case 0: 
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mPaint.setColor(Color. BLUE); 


break; 


case 1: 


mPaint.setColor(Color.GREEN); 


break; 
Case 2: 


mPaint.setColor(Color.RED); 


break; 
Case 3: 


mPaint.setColor(Color. YELLOW); 


break; 
default: 


mPaint.setColor(Color. WHITE); 


break; 
1 
了 


上 绘制 矩形 -- 后 四 


| 将 详细 讲解 */ 


canvas.drawRect((320-80)/2, y, (320-80)/2+80, y+40, mPaint); 


} 


执行 后 将 在 屏幕 内 
现 闪 烁 效果 ， 如 图 


绘制 一 个 和 矩形， 
10-6 所 示 ; 可 以 通过 键 稻 | 


图 10-6 


闪烁 效果 


10.2.2 SurfaceView 类 


SurfaceView 类 在 游戏 开发 中 有 


并 随 着 线程 的 变化 算 形 
上 的 上 、 下 方向 键 来 移动 矩形 ， 如 图 10-7 所 示 


重 的 地 位 ， 它 对 于 画面 的 控制 有 着 更 大 的 


的 填充 颜色 也 随 之 变化 ， 从 而 实 


o 


碟 轩 和 镶 3:04AM 


图 10-7 上 下 移动 


自由 
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度 。SurfaceView 类 是 一 个 双 绥 冲 机 制 ， 在 开发 游戏 时 会 经 常用 到 。 
1. SurfaceView 类 基础 
一 般 来 说 ， 在 Android 中 开发 ， 复杂 一 点 的 游戏 ， 是 必须 用 SurfaceView 的 。SurfaceView 
提供 了 直接 访问 一 个 可 画图 的 界面 ， 可 以 控制 界面 顶部 的 子 视图 层 


在 Android 中 


发 游戏 , 一 般 来 说 , 或 想 写 一 个 复杂 一 点 的 游戏 , 是 必须 用 到 SurfaceView 


来 开发 的 。SurfaceView 提供 直接 访问 一 个 可 画图 的 界面 ， 可 以 控制 在 界面 顶部 的 子 视图 层 。 
的 继承 类 ， 这 个 视图 里 内 和 藤 了 一 个 专门 用 于 绘制 的 Surface。 你 


SurfaceView 是 视图 (View) 日 
可 以 控制 这 个 Surface 的 格式 和 尺寸 。Surfaceview 控 秆 


形 系统 


一 个 


| 这 个 Surface 的 绘 


每 个 surface 创建 一 个 Canvas 对 象 〈 属 性 时 常 改 变 )， 用 来 管理 view 


使 用 


视 ， 格 式 如 下 


/ 


ee 


o 


在 surface 的 大 小 发 生 改 变 时 激发 


制 位 置 。Android 


要 的 概念 和 线索 是 surface。View 及 其 子 类 (如 TextView, Button ) 。 


在 surface 上 的 绘 


点 、 男 线 。 还 要 注意 的 是 ， 使 用 它 的 时 候 ， 一 般 都 是 出 现在 最 顶层 的 。 
的 SurfaceView 的 时 候 , 一 般 情 况 下 还 要 对 其 进行 创建 、 销 毁 及 改变 时 的 情况 进行 监 


public void surfaceChanged(SurfaceHolder holder,int format,int width,int height){} 


// 


在 创 


建 时 激发 ， 一 般 在 这 里 调用 画图 的 线程 


public void surfaceCreated(SurfaceHolder holder){} 
// 销 毁 时 激发 ， 一 般 在 这 里 将 画图 的 线程 停止 、 释 放 
public void surfaceDestroyed(SurfaceHolder holder) {} 


surfaceCreated 会 先 被 调用 ， 然 后 是 surfaceChanged， 当 程序 结束 时 会 调用 


由于 SurfaceHolder 是 一 个 共享 资源 ， 因 此 在 对 其 操作 时 都 应 该 实行 “ 互 斥 操作 ” 即 需 
要 使 用 synchronized 进行 “封锁 ”机 和 伟 
泻 染 文字 的 工作 实际 上 是 
于 工作 线程 LunarThread， 因 


ds 


o 


主线 程 (也 就 是 LunarView 类 ) 的 父 类 View 的 工作 ,而 并 不 属 
此 在 工作 线程 中 是 无 法 控制 的 。 所 以 我 们 改 为 向 主线 程 发 送 一 个 


surfaceDestroyed 


消息 来 代替 ， 让 主线 程 通过 Handler 对 接收 到 的 消息 进行 处 理 ， 从 而 更 新 界面 文字 信息 。 


下 面 将 通 
示 的 流程 
具体 代码 如 下 
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过 一 个 具体 实例 


的 实现 , 介绍 在 Android 中 使 用 SurfaceView 类 实现 屏幕 更 新 显 


o 


E。 本 实例 的 源 代码 保存 在 “光盘 :\daima\10\example8” 其 主 文 伯 


public class example8 extends SurfaceView 


{ 


// 控 制 循 环 


F 是 example8.java, 


implements SurfaceHolder.Callback,Runnable 


boolean mbLoop = false; 


// 定 义 SurfaceHolder 对 象 
SurfaceHolder mSurfaceHolder = null; 


Int miCount = 0; 
Int y= 50; 


第 10 章 ， 典 型 手机 游戏 应 用 
public example8(Context context) 


{ 


super(context); 


// 实例 化 SurfaceHolder 
mSurfaceHolder = this.getHolder(); 


// 添加 回调 
mSurfaceHolder.addCallback(this); 
this.setFocusable(true); 


mbLoop = true; 


/ 在 surface 的 大 小 发 生 改变 时 激发 
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) 


{ 


// 在 surface 创建 时 激发 
public void surfaceCreated(SurfaceHolder holder) 
{ 


/开局 绘图 线程 
new Thread(this).start(); 


/ 在 surface 销毁 时 激发 
public void surfaceDestroyed(SurfaceHolder holder) 


{ 
// 停止 循环 
mbLoop = false; 
} 
/ 绘图 循环 
public void run() 
{ 
while (mbLoop) 
{ 
try 
{ 
Thread.sleep(200); 
} 
catch (Exception e) 
{ 
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} 
synchronized( mSurfaceHolder ) 
{ 
Draw(); 
} 
} 
} 
// 绘图 方法 
public void Draw() 
{ 


/锁定 画布 ， 得 到 canvas 
Canvas canvas= mSurfaceHolder.lockCanvas(); 


if (mSurfaceHolder==null || canvas == null ) 


{ 

return; 
} 
if (miCount < 100) 
{ 

ImiCount++; 
} 
else 
4 

miCount = 0; 
} 
// 绘图 


Paint mPaint = new Paint(); 
mPaint.setAntiAlias(true); 
mPaint.setColor(Color.BLACK); 
/绘制 矩形 -- 清 屏 作 用 
canvas.drawRect(0, 0, 320, 480, mPaint); 
switch (miCount % 4) 
{ 
case 0: 
mPaint.setColor(Color.BLUE); 
break; 
case 1: 
mPaint.setColor(Color.GREEN); 
break; 
case 2: 
mPaint.setColor(Color.RED); 
break; 
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case 3: 
mPaint.setColor(Color. YELLOW); 


break; 
default: 
mPaint.setColor(Color. WHITE); 


break; 


} 


/ 绘制 矩形 -- 后 面 将 详细 讲解 
canvas.drawCircle((320 - 25) /2, y, 50, mPaint); 
/ 绘制 后 解锁 ， 绘 制 后 必须 解锁 才能 显示 
mSurfaceHolder.unlockCanvasAndPost(canvas); 


} 
执行 后 将 在 屏幕 内 绘制 一 个 圆 形 ， 并 随 着 线程 的 变化 圆 形 的 填充 颜色 也 随 之 变化 ， 从 
而 实现 闪烁 效果 ， 如 图 10-8 所 示 ; 可 以 通过 键盘 上 的 上 、 下 方向 键 来 移动 圆 形 ， 如 网 10-9 
所 示 。 


动 剖 全 8:19 AM 动 励 国 3:19 AM 


图 10-8 ”闪烁 效果 图 10-9 上 下 移动 


2. 双 缓冲 

双 缓 冲 的 核心 是 先 通过 setBitmap 方法 将 要 绘制 的 所 有 图 形 绘制 到 一 个 Bitmap 上 ， 然 后 
调用 drawBitmap 方法 绘制 出 这 个 Bitmap， 并 在 屏幕 上 显示 出 来 。 

下 面 将 通过 一 个 具体 实例 的 实现 , 介绍 在 Android 中 使 用 SurfaceView 类 实现 双 绥 冲 的 流 
早 。 本 实例 的 源 代 码 保存 在 “光盘 \daima\10\example6”， 其 主 文件 是 example6.java， 有 具体 代 


Eo 


人 码 如 下 。 


= 


/* 声明 Bitmap 对 象 */ 
Bitmap mBitQQ =null; 


Paint mPaint= null; 


庆 创建 一 个 缓冲 区 * 
Bitmap mSCBitmap = null; 


/* 创建 Canvas 对 象 */ 
Canvas mCanvas = null; 
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public example6(Context context) 


{ 


super(context); 


席 加 载 资源 */ 
mBitQQ = ((BitmapDrawable) getResources().getDrawable(R.drawable.qq)).getBitmap(); 


庆 创建 屏幕 大 小 的 缓冲 区 */ 
mSCBitmap=Bitmap.createBitmap(320, 480, Config.ARGB 8888); 


/# 创建 Canvas */ 


mCanvas = new Canvas(); 


人 # 设置 将 内 容 绘 制 在 mSCBitmap 上 */ 
mCanvas.setBitmap(mSCBitmap); 


mpPaint = new Paint(); 


上 将 mBitQQ 绘制 到 mSCBitmap 上 */ 
mCanvas.drawBitmap(mBitQQ, 0, 0, mPaint); 


席 开启 线程 y 
new Thread(this).start(); 


public void onDraw(Canvas canvas) 


{ 


super.onDraw(canvas); 


上 将 mSCBitmap 显示 到 屏幕 上 */ 
canvas.drawBitmap(mSCBitmap, 0, 0, mPaint); 


和 # 触 屏 事件 
public boolean onTouchEvent(MotionEvent event) 


{ 


return true; 


/# 按键 按 下 事件 沁 
public boolean onKeyDown(int keyCode, KeyEvent event) 


{ 


return true; 
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上 # 按键 弹 起 事件 *% 
public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 


return false; 


public boolean onKeyMultiple(int keyCode, int repeatCount, KeyEvent event) 


{ 
return true; 
} 
1 米 米 
* 线程 处 理 
塌 
public void run() 
{ 
while (!'Thread.currentThread().isInterrupted()) 
{ 
try 
{ 
Thread.sleep(100); 
} 
catch (InterruptedException e) 
{ 
Thread.currentThread().interrupt(); 
} 
// 使 用 postInvalidate 可 以 直接 在 线程 中 更 新 界面 
postInvalidate(); 
} 
} 


} 
执行 后 的 效果 如 图 10-10 所 示 。 


example6 


Te hy 
ww 


图 10-10 ”执行 效果 
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ol 


在 Android 平台 中 提供 了 2 类 动画 ， 分 别 是 Tween 动画 和 Frame 动画 。Tween 动画 用 于 


对 场景 里 的 图 


像 进行 变换 来 产生 动画 效果 ; Frame 动画 用 于 顺序 播放 事先 做 好 的 图 像 。 在 本 


节 的 内 容 
b 


将 通过 具体 实例 的 实现 ， 来 详细 讲解 上 述 2 种 动画 的 实现 流程 。 


10.3.1 ” Tween 动画 


下 面 将 通 
例 的 源 代码 保 


过 一 个 具体 实例 的 实现 过 程 ， 介 绍 在 Android 中 使 用 Tween 动画 的 流程 。 本 实 
存在 “光盘 :daimaN\10\example9”， 其 主 文件 是 example9.java， 具 体 代 码 如 下 。 


public class example9 extends View 


{ 
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/# 定义 Alpha 动画 */ 
private Animation mAnimationAlpha = null; 


/# 定义 Scale 动画 */ 
private Animation mAnimationScale = null; 


/* 定义 Translate 动画 */ 
private Animation mAnimationTranslate = null; 


/* 定义 Rotate 动画 */ 
private Animation mAnimationRotate = null; 


/* 定义 Bitmap 对 象 */ 
Bitmap mBitQQ = null; 


public example9 (Context context) 


{ 
super(context); 
入 加 载 资源 * 
mBitQQ = ((BitmapDrawable) getResources().getDrawable(R.drawable.qq)).getBitmap(); 
} 
public void onDraw(Canvas canvas) 
{ 
super.onDraw(canvas); 
启 绘制 图 片 光 
canvas.drawBitmap(mBitQQ, 0, 0, null); 
} 


第 10 章 _ 典 型 手机 游戏 应 用 “二 
public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 
switch ( keyCode ) 
{ 
case KeyEvent.KEYCODE DPAD UP: 
谍 创建 Alpha 动画 */ 
mAnimationAlpha = new AlphaAnimation(0.1 1.0); 
谍 设置 动画 的 时 间 冯 
mAnimationAlpha.setDuration(3000); 
记 开始 播放 动画 */ 
this.startAnimation(mAnimationAlpha); 
break; 
case KeyEvent.KEYCODE DPAD DOWN: 
雍 创建 Scale 动画 */ 
mAnimationScale =new ScaleAnimation(0.0f, 1.0f, 0.0f, 1.0f, 
Animation.RELATIVE TO SELF., 0.5f, 
Animation.RELATIVE TO SELF, 0.5); 


诺 设置 动画 的 时 间 六 
mAnimationScale.setDuration(500); 
全 开始 播放 动画 */ 
this.startAnimation(mAnimationScale); 
break; 
case KeyEvent.KEYCODE DPAD LEFT: 
/# 创建 Translate 动画 */ 
mAnimationTranslate = new TranslateAnimation(10, 100,10, 100); 
谨 设置 动画 的 时 间 光 
mAnimationTranslate.setDuration(1000); 
从 开始 播放 动画 */ 
this.startAnimation(mAnimationTranslate); 
break; 
case KeyEvent.KEYCODE DPAD RIGHT: 
人/# 创建 Rotate 动画 */ 
mAnimationRotate=new RotateAnimation(0.0f, +360.0f, 
Animation.RELATIVE TO SELF.,0.5f, 
Animation.RELATIVE TO_SELF, 0.5); 


谨 设置 动画 的 时 间 光 
mAnimationRotate.setDuration(1000); 
从 开始 播放 动画 */ 
this.startAnimation(mAnimationRotate); 
break; 


} 


return true; 


} 


程序 执行 后 ， 将 会 把 指定 的 目标 图 片 模拟 为 动画 显示 ， 如 图 10-11 所 示 。 
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吗 园区 阐 怨 3:10rv 


example9 


Se 


图 10-11 执行 效果 


10.3.2 “Frame 动画 


下 面 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 中 使 用 Frame 动画 的 流程 。 本 实例 的 
源 代码 保存 在 “光盘 :\daima\l10\example10”， 其 主 文件 是 example10.java， 具 体 代 码 如 下 。 


/* 定义 AnimationDrawable 动画 */ 
private AnimationDrawable frameAnimation = null; 


Context mContext = null; 


/定义 一 个 Drawable 对 象 */ 
Drawable mBitAnimation = null; 
public example10(Context context) 


{ 


super(context); 
mContext = context; 


访 实例 化 AnimationDrawable 对 象 */ 
frameAnimation =new AnimationDrawable(); 


入 装载 资源 */ 

族 这 里 用 一 个 循环 了 装载 所 有 名 字 类 似 的 资源 */ 
庆 如 “al..…..15.png” 的 图 片 */ 

雍 这 个 方法 用 处 非常 大 */ 

for (inti= 1;i<= 15;1i++) 


\ 


int id = getResources().getldentifier("a" + 1, "drawable", mContext.getPackageName()); 
mBitAnimation = getResources().getDrawable(id); 

庆 为 动画 添加 一 帧 光 

/# 参数 mBitAnimation 是 该 帧 的 图 片 */ 

庆 参数 500 是 该 帧 显示 的 时 间 ， 按 坚 秒 计算 */ 
frameAnimation.addFrame(mBitAnimation, 500); 


/* 设置 播放 模式 是 否 循环 ，false 表示 循环 ，true 表示 不 循环 */ 
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frameAnimation.setOneShot( false ); 


雍 设置 本 类 将 要 显示 这 个 动画 汶 


this.setBackgroundDrawable(frameAnimation); 


public void onDraw(Canvas canvas) 


{ 
super.onDraw(canvas); 
} 
public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 
switch ( keyCode ) 
{ 
case KeyEvent.KEYCODE DPAD UP: 
诺 开始 播放 动画 */ 
frameAnimation.start(); 
break; 
} 
return true; 
} 


} 
实例 执行 后 ， 通 过 按 下 鲍 


图 10-12 执行 效果 


国 563 


Android 开发 完全 实战 宝典 


(ss 


手机 游戏 是 指 运行 于 手机 上 的 游戏 软件 。 目 前 用 来 编写 手机 游戏 , 使 用 最 多 的 语言 是 Java 
语言 。 其 次 是 C 语言 。 随 着 科技 的 发 展 ， 现 在 手机 的 功能 也 越 来 越 多 ， 越 来 越 强 大 。 手 机 游 
戏 也 远 远 不 是 我 们 印象 中 的 “俄罗斯 方块 ””“ 贪 吃 蛇 ” 这 类 画面 简陋 ， 规 则 简单 的 游戏 ， 而 
是 发 展 到 了 可 以 和 掌上 游戏 机 絮 美 ， 有 具有 很 强 的 娱乐 性 和 交互 性 的 复杂 形态 了 。 于 是 ， 抛 弃 
你 的 随身 听 和 Gameboy， 买 一 个 好 手机 吧 ， 你 会 发 现 ， 一 个 手机 已 经 可 以 满足 你 大 部 分 娱乐 

在 本 节 的 内 容 中 ， 将 通过 一 个 具体 实例 的 实现 ， 介 绍 在 Android 平台 中 开发 “ 魔 塔 ” 游 
戏 的 实现 流程 。 本 实例 的 源 代码 保存 在 “ 光 可 :\daima\l0\examplel11”。 


10.4.1 ” Java 游戏 开发 流程 
一 款 J2ME 典型 游戏 的 开发 流程 如 图 10-13 所 示 。 


策划 、 美 工 、 程 序 三 方 会 谈 


策划 、 美 工 、 程 序 三 方 
会 谈 ， 确 定 游戏 内 容 


微调 与 测试 


图 10-13 型 Java 游戏 开发 流程 


1. 立项 

在 制作 游戏 之 前 ， 策 划 首 先 要 确定 一 点 : 到 底 想 要 制作 一 个 什么 样 的 游戏 ? 而 制作 一 个 
游戏 并 不 是 闭门造车 ， 一 个 策划 说 了 就 算数 的 简单 事情 。 制 作 一 款 游 戏 受 到 多 方面 的 限制 。 

1) 市 场 : 即将 开发 的 游戏 是 不 是 具备 市 场 潜力 ? 在 市 场 上 推出 以 后 会 不 会 被 大 家 所 接 
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受 ? 是 否 能 够 取得 良好 的 市 场 回报 ? 

2) 技术 : 即将 开发 的 游戏 从 程序 和 美术 上 是 不 是 完全 能 够 实现 ? 如果 不能 实现 ， 是 不 是 
能 够 有 折 中 的 办 法 ? 

3) 规模 : 以 现 有 的 资源 是 否 能 很 好 的 协调 并 完成 即将 要 开发 的 游戏 ? 是 否 需要 另外 增加 


人 员 或 设备 ? 


4) 周期 : 游戏 的 开发 周期 长 短 是 否 合适 ? 能 否 在 开发 结束 时 正好 赶 上 游戏 的 销售 旺季 ? 


5) 产品 : 


全 
即将 做 的 游戏 在 其 同类 产品 中 是 否 有 新 颖 的 设计 ? 是 否 能 有 吸引 玩家 的 地 方 ? 


如 果 在 游戏 设计 上 达 不 到 创新 ， 是 否 能 够 在 美术 及 程序 方面 加 以 补足 ? 如 果 同 类 型 的 游戏 市 


场 上 已 经 有 了 很 多 ， 那 么 即将 做 的 游戏 的 卖点 在 哪里 ? 
以 上 各 个 问题 都 是 要 经 过 开发 组 全 体 成 员 反 复 进行 讨论 才能 够 确定 下 来 的 ， 集 思 广 益 ， 


共同 探讨 一 个 可 


行 的 方案 。 如 果 对 上 述 全 部 问题 都 能 够 有 肯定 的 答案 的 话 ， 那 么 这 个 项 目 基 


本 是 可 行 的 。 但 是 即便 项 目 获 得 了 通过 ， 在 进行 过 程 中 也 可 能 会 有 种 种 不 可 预知 的 因素 导致 
意外 情况 的 发 生 ， 所 以 项 目 能 够 成 立 ， 上 只 是 游戏 制作 的 开始 。 


2. 大纲 策 


游戏 大 纲 关 系 到 游戏 的 整体 面貌 ， 当 大 纲 集 划 案 定稿 以 后 ， 没 有 特殊 的 情况 ， 是 不 允许 
进行 更 改 的 。 程 序 和 美术 工作 人 员 将 按照 策划 所 构思 的 游戏 形式 来 架构 整个 游戏 ， 因 此 ， 在 
制定 策划 案 时 一 定 要 做 到 慎重 ， 尽 量 考虑 成 熟 。 


3. 游戏 的 


在 项 目 确立 了 以 后 ， 下 一 步 要 进行 的 就 是 进行 游戏 的 大 纲 策划 工作 。 


划 的 进行 


正式 制作 


当 游 戏 大 纲 策划 案 完成 并 讨论 通过 后 ， 游 戏 就 由 三 方面 同时 开始 进行 制作 了 。 在 这 一 阶 
段 ， 策 划 的 主要 任务 是 在 大 纲 的 基础 上 对 游戏 的 所 有 细节 进行 完善 ， 将 游戏 大 纲 逐 步 填充 为 


完整 的 游戏 策划 案 。 根 据 不 同 的 游戏 种 类 ， 所 要 进行 细 化 的 部 分 也 不 尽 相 同 。 
在 正式 制作 的 过 程 中 ， 和 策划、 程序 、 美 工人 员 进 行 及 时 和 经 常 性 的 交流 ， 了 解 工作 进展 


以 及 是 否 有 难以 克服 的 困难 ， 并 且 根 据 现实 情况 有 目的 的 变更 工作 计划 或 设计 思想 。 三 方面 
的 配合 在 游戏 正式 制作 过 程 中 是 最 重要 的 。 


4. 配音 、 


配乐 


在 程序 和 美工 进行 得 差不多 要 结束 的 时 候 ， 就 要 进行 配音 和 配乐 的 工作 了 。 虽 然 音 乐 和 


音效 是 游戏 的 重要 组 成 部 
已 


能 够 起 到 很 好 的 烘托 游戏 气氛 的 作用 ， 但 是 限于 J2ME 游戏 的 


开发 成 本 和 设置 的 处 理 能 


很 好 配合 的 音乐 当 作 游戏 


5. 检测 、 


游戏 刚 制作 完成 ， 在 程序 上 肯定 会 有 很 多 的 错误 ， 错 误 严 重 时 会 导致 游戏 完全 没有 办 法 


分 ， 
力 ， 这 部 份 已 经 被 弱化 到 可 有 可 无 的 地 步 了 。 应 选择 跟 游戏 风格 能 
背景 音乐 ， 这 个 工作 交 给 策划 比较 合适 。 


调试 


进行 下 去 。 同 样 ， 策 划 的 设计 也 会 有 不 完善 的 地 方 ， 主 要 在 游戏 的 参数 部 分 。 参 数 部 分 的 不 
合理 ， 会 影响 游戏 的 可 玩 性 。 此 时 测试 人 员 需 检测 程序 上 的 漏洞 和 通过 试 玩 ， 调 整 游戏 的 各 


个 部 分 参数 使 


基本 平衡 。 


10.4.2 ”设计 游戏 框架 


转 回 正文 ， 


开始 讲解 “ 魔 塔 ”游戏 的 具体 设计 。 因 为 所 有 游戏 是 基于 框架 的 ， 所 以 


设计 一 个 合适 的 框架 尤为 重要 。 为 了 正确 设计 框架 ,我 们 先 看 市 面 上 魔 塔 游戏 的 界面 ， 
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如 图 10-14 所 示 。 

由 游戏 界面 可 知 ， 游 戏 中 包含 了 地 图 、 和 角色、 屏幕 界面 和 道具 等 元 素 ， 上 述 元 素 构成 了 
一 个 视图 ， 例 如 屏幕 视图 、 道 具 视图 和 角色 视图 等 。 


生命 : 1000 上 虹 钥匙 : 100 


图 10-14 游戏 界 国 


1. 界面 视图 

在 Android 中 ， 视 图 是 通过 继承 View 类 实现 的 ， 在 View 类 中 包含 了 各 种 绘制 图 形 的 方 
法 和 事件 ， 这 些 知 识 在 本 章 的 10.1 和 10.2 中 已 经 进行 了 讲解 。 这 样 构建 一 个 游戏 界面 类 将 变 
得 轻而易举 。 界 面 类 GameView 类 的 具体 代码 如 下 。 


package com.examplell; 

import android.content.Context; 

import android.graphics.Canvas; 

import android.view.View; 

public abstract class GameView extends View 
{ 


public GameView(Context context) 


{ 


super(context); 


* (Oparam N/A 


* (Oreturn null 
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protected abstract void onDraw(Canvas canvas); 
/** 
* 按键 按 下 


* 


* (Oparam N/A 


米 


* (Oreturn null 

%/ 
public abstract boolean onKeyDown(int keyCode); 
六 米 米 


* 按键 弹 起 


* 


* (Oparam N/A 


米 


* (Oreturn null 

sp 
public abstract boolean onKeyUp(int keyCode); 
W 米 米 

* 回收 资源 

米 

烛 


protected abstract void reCycle(); 


/ 洲 米 
* 刷新 
加 
/ 
protected abstract void refurbish(); 
} 
屏幕 处 理 


湛 


屏幕 会 伴随 玩家 的 操作 而 变化 ， 所 以 游戏 整体 屏幕 框架 设计 完毕 后 ， 还 需要 设计 屏幕 
的 控制 类 。 在 此 类 中 ， 可 以 根据 不 同 的 游戏 状态 来 设置 屏幕 的 具体 显示 内 容 。 在 此 编写 
MainGame 类 ， 具 体 代码 如 下 


package com.examplell; 


import android.app.Activity; 
import android.content.Context; 


public class MainGame 


{ 
private static GameView m GameView = null; // 当前 需要 显示 的 对 象 
private Context m Context = null; 
private MagicTower m MagicTower = null; 
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private int m_status = -]1; /游戏 状态 
public CMIDIPlayer mCMIDIPlayer; 

public byte mbMusic = 0; 

public MainGame(Context context) 


{ 
m Context = context; 
m MagicTower = (MagicTower)context; 
m status= -1; 
initGame(); 
} 


入 初始 化 游戏 */ 
public void initGame() 


{ 
controlView(yarin.GAME SPLASH); 
mCMIDIPlayer =new CMIDIPlayer(m MagicTower); 
} 
作 得 到 游戏 状态 */ 
public int getStatus() 
{ 
return m _ status; 
} 


入 设置 游戏 状态 */ 
public void setStatus(int status) 
{ 
m status = status; 
} 
作 得 到 主 类 对 象 */ 
public Activity getMagicTower() 
{ 


retum m MagicTower; 


久 得 到 当前 需要 显示 的 对 象 */ 
public static GameView getMainView() 


{ 


retum m GameView; 


谨 控制 显示 什么 界面 */ 
public void controlView(int status) 


{ 


if(m status != status) 


{ 


if(m GameView !=null) 


3. 线程 更 新 
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m GameView.reCycle(); 
System.gc(); 


} 


freeGameView(m GameView); 

switch (status) 

{ 

case yarin.GAME SPLASH: 
m GameView = new SplashScreen(m Context,this); 
break; 

case yarin.GAME MENU: 
m GameView = new MainMenu(m Context,this); 
break; 

case yarin.GAME HELP: 
m GameView = new HelpScreen(m Context,this); 
break; 

case yarin.GAME ABOUT: 
m GameView= new AboutScreen(m Context,this); 
break; 

case yarin.GAME RUN: 
m GameView= new GameScreen(m Context,m MagicTower,this,true); 
break; 

case yarin.GAME CONTINUE: 
m GameView = new GameScreen(m Context,m MagicTower,this,false); 
break; 

} 


setStatus(status); 


/* 释放 界面 对 象 */ 


public void freeGameView(GameView gameView) 


{ 


} 


游戏 天 


来 进行 游戏 更 新 。 此 更 新 能 在 文件 ThreadCanvas.java 中 实现 ， 具 体 代码 如 下 。 


屏幕 界面 设计 告 一 段落， 但 是 要 实现 界 孟 
线程 ， 并 通过 getMainView0 方 法 来 获取 当前 显示 的 界面 ， 并 根据 7 


F 启 一 个 主 


if(gameView != null) 


{ 
gameView = null; 
System.gc(); 


pa 


[的 真正 更 新 ， 则 需要 线程 来 实现 。 在 此 可 以 为 


\ 同 的 界面 
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package com.examplell; 


import android.content.Context; 
import android.graphics.Canvas; 


import android.util.Log; 
import android.view.View; 


public class ThreadCanvas extends View implements Runnable 


{ 
private String m Tag = "ThreadCanvas Tag"; 


public ThreadCanvas(Context context) 


{ 
super(context); 

} 
Ya * 

* 绘图 

水 

* (Oparam N 

/A 


* 


* (Oreturn null 


光 
protected void onDraw(Canvas canvas) 
{ 
if (MainGame.getMainView() != null) 
{ 
MainGame.getMainView().onDraw(canvas); 
} 
else 
{ 
Log.i(m Tag, "null"); 
} 
} 
/** 


让 
public void start() 
{ 
Thread t = new Thread(this); 
57O 厦 下 
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MainGame.getMainView().onKeyDown(keyCode); 


else 


Log.i(m Tag, "null"); 
} 


return true; 


此 按键 弹 起 */ 
boolean onKeyUp(int keyCode) 


{ 
if (MainGame.getMainView() != null) 
{ 
MainGame.getMainView().onKeyUp(keyCode); 
} 
else 
{ 
Log.i(m Tag, "null"); 
} 
return true; 
} 


} 


4. 具体 显示 

经 过 上 述 类 的 设计 后 ， 还 需要 进行 最 后 一 步 ， 即 用 一 个 Activity 来 显示 具体 的 界面 。 因 
为 是 在 ThreadCanvas 中 控制 界面 的 ， 所 以 只 需 用 setContentView 来 显示 一 个 ThreadCanvas 对 
象 即 可 。 上 述 功能 通过 文件 MagicTowerjava 实现 ， 具 体 代码 如 下 。 


package com.examplell; 


import android.app.Activity; 

import android.os.Bundle; 

import android.view.KeyEvent; 

import android.view.Window; 

import android.view.WindowManager; 


public class MagicTower extends Activity 


{ 


private ThreadCanvas mThreadCanvas = null; 


public void onCreate(Bundle savedInstanceState) 


{ 
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super.onCreate(savedInstanceState); 

setTheme(android.R.style.Theme Black NoTitleBar Fullscreen); 

requestWindowFeature(Window.FEATURE NO TITLE); 

getWindow().setFlags(Window Manager.LayoutParams.FLAG FULLSCREEN, WindowManager. 
LayoutParams.FLAG FULLSCREEN); 


new MainGame(this); 
mThreadCanvas = new ThreadCanvas(this); 


setContentView(mThreadCanvas); 


/** 
* 暂停 
米 
* (Oparam N 
/A 


米 


* (Oreturn null 
/| 
protected void onPause() 


{ 


super.onPause(); 


* 
4 
由 
\ 
NH> 


* (Oparam N 
/A 
* 
* (Oreturn null 
protected void onResume() 
{ 
super.onResume(); 
mThreadCanvas.requestFocus(); 
mThreadCanvas.start(); 


A 
* 按键 按 下 
水 
* (Oparam N 
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时 /A 
米 
* (Oreturn null 
public boolean onKeyDown(int keyCode, KeyEvent event) 


{ 
mThreadCanvas.onKeyDown(keyCode); 
return false; 


7 米 米 

* 按键 弹 起 

米 

* (Oparam N 

yi /A 

米 

* (Oreturn null 

he 
public boolean onKeyUp(int keyCode, KeyEvent event) 
{ 


mThreadCanvas.onKeyUp(keyCode); 
return false; 


} 
通过 上 述 处 理 ， 一 个 基本 的 游戏 框架 建设 完毕 。 在 后 续 的 开发 中 ， 只 需 直 接 继 承 上 面 的 


类 界面 ， 即 继承 GameView， 然 后 在 MainView 中 更 改 游戏 状态 即 可 。 因 为 本 书 篇 幅 有 限 ， 后 
续 代码 读者 可 以 参考 本 书 配 套 的 源 代 码 。 
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本 书 详细 讲解 了 Android 技 术 在 各 个 领域 的 具体 应 用 ， 介 绍 了 各 个 实例 的 具体 实 
现 过程 。 全 书 分 为 10 章 ， 第 1、2 章 是 基础 知识 ， 讲 解 了 Android 前 景 和 搭建 开发 环境 
的 过 程 ， 第 3 章 详 细 讲 解 了 Android 在 人 机 交互 界面 领域 典型 实例 的 设计 过 程 ， 第 4 章 
详细 讲解 了 Android 各 个 组 件 的 使 用 方法 ; 第 5 章 讲解 了 Android 在 交互 式 应 用 领域 的 
具体 应 用 ， 第 6 章 讲解 了 Android 在 手机 自动 服务 领域 中 的 应 用 ; 第 7 章 讲解 了 Android 
在 娱乐 和 多 媒体 领域 的 具体 应 用 ， 第 8 章 讲解 了 Android 在 互联 网 领域 各 个 实例 的 实现 
过 程 ， 第 9 章 讲解 了 Android 在 官方 服务 绑 定 领域 各 个 实例 的 实现 过 程 ; 第 10 章 讲解 了 


Android 在 绘图 和 游戏 开发 领域 的 具体 应 用 。 


本 书 适用 于 Android 的 初 、 中 、 高 级 用 户 ， 既 可 以 作为 初学 者 的 自学 手册 ， 也 可 


以 作为 有 一 定 程序 开发 基础 人 员 的 参考 书 。 
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